2017-11-01 20:39:49 +01:00
///////////////////////////////////////////////////////////////////////////////
// Copyright (C) 2017, Tadas Baltrusaitis, all rights reserved.
//
// ACADEMIC OR NON-PROFIT ORGANIZATION NONCOMMERCIAL RESEARCH USE ONLY
//
// BY USING OR DOWNLOADING THE SOFTWARE, YOU ARE AGREEING TO THE TERMS OF THIS LICENSE AGREEMENT.
// IF YOU DO NOT AGREE WITH THESE TERMS, YOU MAY NOT USE OR DOWNLOAD THE SOFTWARE.
//
// License can be found in OpenFace-license.txt
//
// * Any publications arising from the use of this software, including but
// not limited to academic journal and conference publications, technical
// reports and manuals, must cite at least one of the following works:
//
// OpenFace: an open source facial behavior analysis toolkit
// Tadas Baltru<72> aitis, Peter Robinson, and Louis-Philippe Morency
// in IEEE Winter Conference on Applications of Computer Vision, 2016
//
// Rendering of Eyes for Eye-Shape Registration and Gaze Estimation
// Erroll Wood, Tadas Baltru<72> aitis, Xucong Zhang, Yusuke Sugano, Peter Robinson, and Andreas Bulling
// in IEEE International. Conference on Computer Vision (ICCV), 2015
//
// Cross-dataset learning and person-speci?c normalisation for automatic Action Unit detection
// Tadas Baltru<72> aitis, Marwa Mahmoud, and Peter Robinson
// in Facial Expression Recognition and Analysis Challenge,
// IEEE International Conference on Automatic Face and Gesture Recognition, 2015
//
// Constrained Local Neural Fields for robust facial landmark detection in the wild.
// Tadas Baltru<72> aitis, Peter Robinson, and Louis-Philippe Morency.
// in IEEE Int. Conference on Computer Vision Workshops, 300 Faces in-the-Wild Challenge, 2013.
//
///////////////////////////////////////////////////////////////////////////////
# include "RecorderOpenFace.h"
// For sorting
# include <algorithm>
// File manipulation
# include <fstream>
# include <sstream>
# include <iostream>
// Boost includes for file system manipulation
# include <filesystem.hpp>
# include <filesystem/fstream.hpp>
# include <boost/algorithm/string.hpp>
using namespace boost : : filesystem ;
2017-11-08 10:02:50 +01:00
using namespace Utilities ;
2017-11-01 20:39:49 +01:00
2017-11-03 09:34:55 +01:00
# define WARN_STREAM( stream ) \
std : : cout < < " Warning: " < < stream < < std : : endl
2017-11-02 10:06:53 +01:00
void CreateDirectory ( std : : string output_path )
2017-11-01 20:39:49 +01:00
{
// Creating the right directory structure
auto p = path ( output_path ) ;
if ( ! boost : : filesystem : : exists ( p ) )
{
bool success = boost : : filesystem : : create_directories ( p ) ;
if ( ! success )
{
std : : cout < < " Failed to create a directory... " < < p . string ( ) < < std : : endl ;
}
}
}
2017-11-13 10:07:52 +01:00
RecorderOpenFace : : RecorderOpenFace ( const std : : string in_filename , RecorderOpenFaceParameters parameters , std : : vector < std : : string > & arguments ) : video_writer ( ) , params ( parameters )
2017-11-01 20:39:49 +01:00
{
// From the filename, strip out the name without directory and extension
filename = path ( in_filename ) . replace_extension ( " " ) . filename ( ) . string ( ) ;
2017-11-09 19:37:26 +01:00
2017-11-12 22:40:30 +01:00
// Consuming the input arguments
bool * valid = new bool [ arguments . size ( ) ] ;
for ( size_t i = 0 ; i < arguments . size ( ) ; + + i )
{
valid [ i ] = true ;
}
string record_root ;
for ( size_t i = 0 ; i < arguments . size ( ) ; + + i )
{
if ( arguments [ i ] . compare ( " -outroot " ) = = 0 )
{
record_root = arguments [ i + 1 ] ;
}
}
2017-11-09 19:37:26 +01:00
// Determine output directory
2017-11-12 22:40:30 +01:00
bool output_found = false ;
2017-11-09 19:37:26 +01:00
for ( size_t i = 0 ; i < arguments . size ( ) ; + + i )
{
if ( arguments [ i ] . compare ( " -out_dir " ) = = 0 )
{
2017-11-12 22:40:30 +01:00
record_root = ( boost : : filesystem : : path ( record_root ) / boost : : filesystem : : path ( arguments [ i + 1 ] ) ) . string ( ) ;
}
else if ( ! output_found & & arguments [ i ] . compare ( " -of " ) = = 0 )
{
record_root = ( boost : : filesystem : : path ( record_root ) / boost : : filesystem : : path ( arguments [ i + 1 ] ) ) . remove_filename ( ) . string ( ) ;
filename = path ( boost : : filesystem : : path ( arguments [ i + 1 ] ) ) . replace_extension ( " " ) . filename ( ) . string ( ) ;
valid [ i ] = false ;
valid [ i + 1 ] = false ;
i + + ;
output_found = true ;
}
}
for ( int i = ( int ) arguments . size ( ) - 1 ; i > = 0 ; - - i )
{
if ( ! valid [ i ] )
{
arguments . erase ( arguments . begin ( ) + i ) ;
2017-11-09 19:37:26 +01:00
}
}
2017-11-01 20:39:49 +01:00
// Construct the directories required for the output
2017-11-02 10:06:53 +01:00
CreateDirectory ( record_root ) ;
2017-11-01 20:39:49 +01:00
2017-11-12 22:40:30 +01:00
// Create the filename for the general output file that contains all of the meta information about the recording
path of_det_name ( filename ) ;
of_det_name = path ( record_root ) / of_det_name . concat ( " _of_details.txt " ) ;
// Write in the of file what we are outputing what is the input etc.
metadata_file . open ( of_det_name . string ( ) , std : : ios_base : : out ) ;
2017-11-14 21:19:11 +01:00
if ( ! metadata_file . is_open ( ) )
{
cout < < " ERROR: could not open the output file: " < < of_det_name < < " , either the path of the output directory is wrong or you do not have the permissions to write to it " < < endl ;
exit ( 1 ) ;
}
2017-11-12 22:40:30 +01:00
// Populate the metadata file
metadata_file < < " Input: " < < in_filename < < endl ;
2017-11-01 20:39:49 +01:00
// Create the required individual recorders, CSV, HOG, aligned, video
2017-11-05 08:45:19 +01:00
csv_filename = ( path ( record_root ) / path ( filename ) . replace_extension ( " .csv " ) ) . string ( ) ;
2017-11-12 22:40:30 +01:00
metadata_file < < " Output csv: " < < csv_filename < < endl ;
2017-11-01 20:39:49 +01:00
// Consruct HOG recorder here
2017-11-03 22:35:55 +01:00
if ( params . outputHOG ( ) )
2017-11-01 21:03:31 +01:00
{
std : : string hog_filename = ( path ( record_root ) / path ( filename ) . replace_extension ( " .hog " ) ) . string ( ) ;
hog_recorder . Open ( hog_filename ) ;
2017-11-12 22:40:30 +01:00
metadata_file < < " Output HOG: " < < csv_filename < < endl ;
2017-11-01 21:03:31 +01:00
}
2017-11-01 20:39:49 +01:00
2017-11-05 08:45:19 +01:00
// saving the videos
2017-11-03 22:35:55 +01:00
if ( params . outputTrackedVideo ( ) )
2017-11-03 09:34:55 +01:00
{
this - > video_filename = ( path ( record_root ) / path ( filename ) . replace_extension ( " .avi " ) ) . string ( ) ;
2017-11-13 10:07:52 +01:00
if ( parameters . isSequence ( ) )
{
metadata_file < < " Output video: " < < this - > video_filename < < endl ;
}
else
{
metadata_file < < " Output image: " < < this - > video_filename < < endl ;
}
2017-11-03 09:34:55 +01:00
}
2017-11-05 09:51:27 +01:00
// Prepare image recording
if ( params . outputAlignedFaces ( ) )
{
aligned_output_directory = ( path ( record_root ) / path ( filename + " _aligned " ) ) . string ( ) ;
CreateDirectory ( aligned_output_directory ) ;
2017-11-12 22:40:30 +01:00
metadata_file < < " Output aligned directory: " < < this - > aligned_output_directory < < endl ;
2017-11-05 09:51:27 +01:00
}
2017-11-12 22:40:30 +01:00
2017-11-02 21:11:58 +01:00
observation_count = 0 ;
2017-11-01 20:39:49 +01:00
}
2017-11-05 09:51:27 +01:00
void RecorderOpenFace : : SetObservationFaceAlign ( const cv : : Mat & aligned_face )
{
this - > aligned_face = aligned_face ;
}
2017-11-03 17:40:07 +01:00
void RecorderOpenFace : : SetObservationVisualization ( const cv : : Mat & vis_track )
2017-11-03 09:34:55 +01:00
{
2017-11-03 22:35:55 +01:00
if ( params . outputTrackedVideo ( ) )
2017-11-03 09:34:55 +01:00
{
// Initialize the video writer if it has not been opened yet
if ( ! video_writer . isOpened ( ) )
{
std : : string video_filename = ( path ( record_root ) / path ( filename ) . replace_extension ( " .avi " ) ) . string ( ) ;
2017-11-03 22:35:55 +01:00
std : : string output_codec = params . outputCodec ( ) ;
2017-11-03 09:34:55 +01:00
try
{
2017-11-04 21:57:24 +01:00
video_writer . open ( video_filename , CV_FOURCC ( output_codec [ 0 ] , output_codec [ 1 ] , output_codec [ 2 ] , output_codec [ 3 ] ) , params . outputFps ( ) , vis_track . size ( ) , true ) ;
2017-11-03 09:34:55 +01:00
}
catch ( cv : : Exception e )
{
WARN_STREAM ( " Could not open VideoWriter, OUTPUT FILE WILL NOT BE WRITTEN. Currently using codec " < < output_codec < < " , try using an other one (-oc option) " ) ;
}
}
vis_to_out = vis_track ;
}
}
2017-11-02 21:11:58 +01:00
void RecorderOpenFace : : WriteObservation ( )
{
observation_count + + ;
// Write out the CSV file (it will always be there, even if not outputting anything more but frame/face numbers)
2017-11-03 21:00:24 +01:00
if ( observation_count = = 1 )
{
2017-11-05 08:45:19 +01:00
// As we are writing out the header, work out some things like number of landmarks, names of AUs etc.
int num_face_landmarks = landmarks_2D . rows / 2 ;
2017-11-12 11:55:47 +01:00
int num_eye_landmarks = ( int ) eye_landmarks2D . size ( ) ;
2017-11-05 10:30:56 +01:00
int num_model_modes = pdm_params_local . rows ;
2017-11-05 08:45:19 +01:00
std : : vector < std : : string > au_names_class ;
for ( auto au : au_occurences )
{
au_names_class . push_back ( au . first ) ;
}
std : : sort ( au_names_class . begin ( ) , au_names_class . end ( ) ) ;
std : : vector < std : : string > au_names_reg ;
for ( auto au : au_intensities )
{
au_names_reg . push_back ( au . first ) ;
}
std : : sort ( au_names_reg . begin ( ) , au_names_reg . end ( ) ) ;
2017-11-07 08:46:25 +01:00
csv_recorder . Open ( csv_filename , params . isSequence ( ) , params . output2DLandmarks ( ) , params . output3DLandmarks ( ) , params . outputPDMParams ( ) , params . outputPose ( ) ,
2017-11-04 21:57:24 +01:00
params . outputAUs ( ) , params . outputGaze ( ) , num_face_landmarks , num_model_modes , num_eye_landmarks , au_names_class , au_names_reg ) ;
2017-11-03 21:00:24 +01:00
}
2017-11-03 09:34:55 +01:00
this - > csv_recorder . WriteLine ( observation_count , timestamp , landmark_detection_success ,
landmark_detection_confidence , landmarks_2D , landmarks_3D , pdm_params_local , pdm_params_global , head_pose ,
2017-11-12 11:55:47 +01:00
gaze_direction0 , gaze_direction1 , gaze_angle , eye_landmarks2D , eye_landmarks3D , au_intensities , au_occurences ) ;
2017-11-02 21:11:58 +01:00
2017-11-04 21:57:24 +01:00
if ( params . outputHOG ( ) )
2017-11-03 17:40:07 +01:00
{
this - > hog_recorder . Write ( ) ;
}
2017-11-03 09:34:55 +01:00
2017-11-05 09:51:27 +01:00
// Write aligned faces
if ( params . outputAlignedFaces ( ) )
{
char name [ 100 ] ;
// Filename is based on frame number
std : : sprintf ( name , " frame_det_%06d.bmp " , observation_count ) ;
// Construct the output filename
boost : : filesystem : : path slash ( " / " ) ;
std : : string preferredSlash = slash . make_preferred ( ) . string ( ) ;
string out_file = aligned_output_directory + preferredSlash + string ( name ) ;
bool write_success = cv : : imwrite ( out_file , aligned_face ) ;
if ( ! write_success )
{
WARN_STREAM ( " Could not output similarity aligned image image " ) ;
}
}
2017-11-04 21:57:24 +01:00
if ( params . outputTrackedVideo ( ) )
2017-11-03 09:34:55 +01:00
{
2017-11-03 10:04:00 +01:00
if ( vis_to_out . empty ( ) )
2017-11-03 09:34:55 +01:00
{
WARN_STREAM ( " Output tracked video frame is not set " ) ;
}
video_writer . write ( vis_to_out ) ;
// Clear the output
vis_to_out = cv : : Mat ( ) ;
}
2017-11-02 21:11:58 +01:00
}
2017-11-02 10:06:53 +01:00
void RecorderOpenFace : : SetObservationHOG ( bool good_frame , const cv : : Mat_ < double > & hog_descriptor , int num_cols , int num_rows , int num_channels )
2017-11-01 21:03:31 +01:00
{
2017-11-02 10:06:53 +01:00
this - > hog_recorder . SetObservationHOG ( good_frame , hog_descriptor , num_cols , num_rows , num_channels ) ;
2017-11-01 21:03:31 +01:00
}
2017-11-02 10:06:53 +01:00
void RecorderOpenFace : : SetObservationTimestamp ( double timestamp )
{
this - > timestamp = timestamp ;
}
void RecorderOpenFace : : SetObservationLandmarks ( const cv : : Mat_ < double > & landmarks_2D , const cv : : Mat_ < double > & landmarks_3D ,
2017-11-03 09:34:55 +01:00
const cv : : Vec6d & pdm_params_global , const cv : : Mat_ < double > & pdm_params_local , double confidence , bool success )
2017-11-02 10:06:53 +01:00
{
this - > landmarks_2D = landmarks_2D ;
this - > landmarks_3D = landmarks_3D ;
2017-11-03 09:34:55 +01:00
this - > pdm_params_global = pdm_params_global ;
this - > pdm_params_local = pdm_params_local ;
2017-11-02 10:06:53 +01:00
this - > landmark_detection_confidence = confidence ;
this - > landmark_detection_success = success ;
}
void RecorderOpenFace : : SetObservationPose ( const cv : : Vec6d & pose )
{
this - > head_pose = pose ;
}
void RecorderOpenFace : : SetObservationActionUnits ( const std : : vector < std : : pair < std : : string , double > > & au_intensities ,
const std : : vector < std : : pair < std : : string , double > > & au_occurences )
{
this - > au_intensities = au_intensities ;
this - > au_occurences = au_occurences ;
}
2017-11-03 09:34:55 +01:00
void RecorderOpenFace : : SetObservationGaze ( const cv : : Point3f & gaze_direction0 , const cv : : Point3f & gaze_direction1 ,
2017-11-11 22:13:29 +01:00
const cv : : Vec2d & gaze_angle , const std : : vector < cv : : Point2d > & eye_landmarks2D , const std : : vector < cv : : Point3d > & eye_landmarks3D )
2017-11-02 10:06:53 +01:00
{
2017-11-03 09:34:55 +01:00
this - > gaze_direction0 = gaze_direction0 ;
this - > gaze_direction1 = gaze_direction1 ;
2017-11-02 10:06:53 +01:00
this - > gaze_angle = gaze_angle ;
2017-11-11 22:13:29 +01:00
this - > eye_landmarks2D = eye_landmarks2D ;
this - > eye_landmarks3D = eye_landmarks3D ;
2017-11-02 10:06:53 +01:00
}
2017-11-03 17:40:07 +01:00
RecorderOpenFace : : ~ RecorderOpenFace ( )
{
this - > Close ( ) ;
}
void RecorderOpenFace : : Close ( )
{
hog_recorder . Close ( ) ;
csv_recorder . Close ( ) ;
video_writer . release ( ) ;
2017-11-12 22:40:30 +01:00
metadata_file . close ( ) ;
2017-11-03 17:40:07 +01:00
}
2017-11-02 10:06:53 +01:00
2017-11-01 20:39:49 +01:00