06_face_from_image.cpp 11.0 KB
Newer Older
1
// ----------------------------- OpenPose C++ API Tutorial - Example 6 - Face from Image -----------------------------
G
gineshidalgo99 已提交
2 3 4 5
// It reads an image and the face location, process it, and displays the face keypoints. In addition,
// it includes all the OpenPose configuration flags.
// Input: An image and the face rectangle locations.
// Output: OpenPose face keypoint detection.
G
Gines Hidalgo 已提交
6
// NOTE: This demo is auto-selecting the following flags: `--body 0 --face --face_detector 2`
G
gineshidalgo99 已提交
7

8 9 10
// Third-party dependencies
#include <opencv2/opencv.hpp>
// Command-line user interface
G
gineshidalgo99 已提交
11 12 13 14 15 16 17 18
#define OPENPOSE_FLAGS_DISABLE_PRODUCER
#define OPENPOSE_FLAGS_DISABLE_DISPLAY
#include <openpose/flags.hpp>
// OpenPose dependencies
#include <openpose/headers.hpp>

// Custom OpenPose flags
// Producer
G
gineshidalgo99 已提交
19
DEFINE_string(image_path, "examples/media/COCO_val2014_000000000241.jpg",
G
gineshidalgo99 已提交
20
    "Process an image. Read all standard formats (jpg, png, bmp, etc.).");
G
gineshidalgo99 已提交
21 22 23
// Display
DEFINE_bool(no_display,                 false,
    "Enable to disable the visual display.");
G
gineshidalgo99 已提交
24 25

// This worker will just read and return all the jpg files in a directory
G
gineshidalgo99 已提交
26
void display(const std::shared_ptr<std::vector<std::shared_ptr<op::Datum>>>& datumsPtr)
G
gineshidalgo99 已提交
27
{
G
gineshidalgo99 已提交
28 29 30 31 32 33 34 35
    try
    {
        // User's displaying/saving/other processing here
            // datum.cvOutputData: rendered frame with pose or heatmaps
            // datum.poseKeypoints: Array<float> with the estimated pose
        if (datumsPtr != nullptr && !datumsPtr->empty())
        {
            // Display image
36
            const cv::Mat cvMat = OP_OP2CVCONSTMAT(datumsPtr->at(0)->cvOutputData);
37 38 39 40 41 42 43
            if (!cvMat.empty())
            {
                cv::imshow(OPEN_POSE_NAME_AND_VERSION + " - Tutorial C++ API", cvMat);
                cv::waitKey(0);
            }
            else
                op::opLog("Empty cv::Mat as output.", op::Priority::High, __LINE__, __FUNCTION__, __FILE__);
G
gineshidalgo99 已提交
44 45
        }
        else
46
            op::opLog("Nullptr or empty datumsPtr found.", op::Priority::High);
G
gineshidalgo99 已提交
47 48
    }
    catch (const std::exception& e)
G
gineshidalgo99 已提交
49
    {
G
gineshidalgo99 已提交
50
        op::error(e.what(), __LINE__, __FUNCTION__, __FILE__);
G
gineshidalgo99 已提交
51 52 53
    }
}

G
gineshidalgo99 已提交
54
void printKeypoints(const std::shared_ptr<std::vector<std::shared_ptr<op::Datum>>>& datumsPtr)
G
gineshidalgo99 已提交
55
{
G
gineshidalgo99 已提交
56 57 58 59 60
    try
    {
        // Example: How to use the pose keypoints
        if (datumsPtr != nullptr && !datumsPtr->empty())
        {
61 62 63 64
            op::opLog("Body keypoints: " + datumsPtr->at(0)->poseKeypoints.toString(), op::Priority::High);
            op::opLog("Face keypoints: " + datumsPtr->at(0)->faceKeypoints.toString(), op::Priority::High);
            op::opLog("Left hand keypoints: " + datumsPtr->at(0)->handKeypoints[0].toString(), op::Priority::High);
            op::opLog("Right hand keypoints: " + datumsPtr->at(0)->handKeypoints[1].toString(), op::Priority::High);
G
gineshidalgo99 已提交
65 66
        }
        else
67
            op::opLog("Nullptr or empty datumsPtr found.", op::Priority::High);
G
gineshidalgo99 已提交
68 69
    }
    catch (const std::exception& e)
G
gineshidalgo99 已提交
70
    {
G
gineshidalgo99 已提交
71
        op::error(e.what(), __LINE__, __FUNCTION__, __FILE__);
G
gineshidalgo99 已提交
72 73 74
    }
}

G
gineshidalgo99 已提交
75
void configureWrapper(op::Wrapper& opWrapper)
G
gineshidalgo99 已提交
76
{
G
gineshidalgo99 已提交
77 78
    try
    {
G
gineshidalgo99 已提交
79
        // Configuring OpenPose
G
gineshidalgo99 已提交
80

G
gineshidalgo99 已提交
81
        // logging_level
82 83 84
        op::checkBool(
            0 <= FLAGS_logging_level && FLAGS_logging_level <= 255, "Wrong logging_level value.",
            __LINE__, __FUNCTION__, __FILE__);
G
gineshidalgo99 已提交
85 86 87
        op::ConfigureLog::setPriorityThreshold((op::Priority)FLAGS_logging_level);
        op::Profiler::setDefaultX(FLAGS_profile_speed);

G
gineshidalgo99 已提交
88 89
        // Applying user defined configuration - GFlags to program variables
        // outputSize
G
Gines Hidalgo 已提交
90
        const auto outputSize = op::flagsToPoint(op::String(FLAGS_output_resolution), "-1x-1");
G
gineshidalgo99 已提交
91
        // netInputSize
G
Gines Hidalgo 已提交
92
        const auto netInputSize = op::flagsToPoint(op::String(FLAGS_net_resolution), "-1x368");
G
gineshidalgo99 已提交
93
        // faceNetInputSize
G
Gines Hidalgo 已提交
94
        const auto faceNetInputSize = op::flagsToPoint(op::String(FLAGS_face_net_resolution), "368x368 (multiples of 16)");
G
gineshidalgo99 已提交
95
        // handNetInputSize
G
Gines Hidalgo 已提交
96
        const auto handNetInputSize = op::flagsToPoint(op::String(FLAGS_hand_net_resolution), "368x368 (multiples of 16)");
97 98
        // poseMode
        const auto poseMode = op::flagsToPoseMode(FLAGS_body);
G
gineshidalgo99 已提交
99
        // poseModel
G
Gines Hidalgo 已提交
100
        const auto poseModel = op::flagsToPoseModel(op::String(FLAGS_model_pose));
G
gineshidalgo99 已提交
101 102
        // JSON saving
        if (!FLAGS_write_keypoint.empty())
103 104 105
            op::opLog(
                "Flag `write_keypoint` is deprecated and will eventually be removed. Please, use `write_json`"
                " instead.", op::Priority::Max);
106 107
        // keypointScaleMode
        const auto keypointScaleMode = op::flagsToScaleMode(FLAGS_keypoint_scale);
G
gineshidalgo99 已提交
108 109 110
        // heatmaps to add
        const auto heatMapTypes = op::flagsToHeatMaps(FLAGS_heatmaps_add_parts, FLAGS_heatmaps_add_bkg,
                                                      FLAGS_heatmaps_add_PAFs);
111
        const auto heatMapScaleMode = op::flagsToHeatMapScaleMode(FLAGS_heatmaps_scale);
G
gineshidalgo99 已提交
112 113
        // >1 camera view?
        const auto multipleView = (FLAGS_3d || FLAGS_3d_views > 1);
G
gineshidalgo99 已提交
114
        // Face and hand detectors
G
gineshidalgo99 已提交
115
        const auto faceDetector = op::flagsToDetector(FLAGS_face_detector);
G
gineshidalgo99 已提交
116
        const auto handDetector = op::flagsToDetector(FLAGS_hand_detector);
G
gineshidalgo99 已提交
117 118
        // Enabling Google Logging
        const bool enableGoogleLogging = true;
G
gineshidalgo99 已提交
119

G
gineshidalgo99 已提交
120 121
        // Pose configuration (use WrapperStructPose{} for default and recommended configuration)
        const op::WrapperStructPose wrapperStructPose{
122
            poseMode, netInputSize, outputSize, keypointScaleMode, FLAGS_num_gpu, FLAGS_num_gpu_start,
G
gineshidalgo99 已提交
123 124
            FLAGS_scale_number, (float)FLAGS_scale_gap, op::flagsToRenderMode(FLAGS_render_pose, multipleView),
            poseModel, !FLAGS_disable_blending, (float)FLAGS_alpha_pose, (float)FLAGS_alpha_heatmap,
G
Gines Hidalgo 已提交
125
            FLAGS_part_to_show, op::String(FLAGS_model_folder), heatMapTypes, heatMapScaleMode, FLAGS_part_candidates,
126
            (float)FLAGS_render_threshold, FLAGS_number_people_max, FLAGS_maximize_positives, FLAGS_fps_max,
G
Gines Hidalgo 已提交
127 128
            op::String(FLAGS_prototxt_path), op::String(FLAGS_caffemodel_path),
            (float)FLAGS_upsampling_ratio, enableGoogleLogging};
G
gineshidalgo99 已提交
129 130 131
        opWrapper.configure(wrapperStructPose);
        // Face configuration (use op::WrapperStructFace{} to disable it)
        const op::WrapperStructFace wrapperStructFace{
G
gineshidalgo99 已提交
132
            FLAGS_face, faceDetector, faceNetInputSize,
G
gineshidalgo99 已提交
133
            op::flagsToRenderMode(FLAGS_face_render, multipleView, FLAGS_render_pose),
G
gineshidalgo99 已提交
134 135 136 137
            (float)FLAGS_face_alpha_pose, (float)FLAGS_face_alpha_heatmap, (float)FLAGS_face_render_threshold};
        opWrapper.configure(wrapperStructFace);
        // Hand configuration (use op::WrapperStructHand{} to disable it)
        const op::WrapperStructHand wrapperStructHand{
G
gineshidalgo99 已提交
138
            FLAGS_hand, handDetector, handNetInputSize, FLAGS_hand_scale_number, (float)FLAGS_hand_scale_range,
G
gineshidalgo99 已提交
139 140 141 142 143 144 145
            op::flagsToRenderMode(FLAGS_hand_render, multipleView, FLAGS_render_pose), (float)FLAGS_hand_alpha_pose,
            (float)FLAGS_hand_alpha_heatmap, (float)FLAGS_hand_render_threshold};
        opWrapper.configure(wrapperStructHand);
        // Extra functionality configuration (use op::WrapperStructExtra{} to disable it)
        const op::WrapperStructExtra wrapperStructExtra{
            FLAGS_3d, FLAGS_3d_min_views, FLAGS_identification, FLAGS_tracking, FLAGS_ik_threads};
        opWrapper.configure(wrapperStructExtra);
146
        // Output (comment or use default argument to disable any output)
G
gineshidalgo99 已提交
147
        const op::WrapperStructOutput wrapperStructOutput{
G
Gines Hidalgo 已提交
148 149 150 151 152 153 154
            FLAGS_cli_verbose, op::String(FLAGS_write_keypoint), op::stringToDataFormat(FLAGS_write_keypoint_format),
            op::String(FLAGS_write_json), op::String(FLAGS_write_coco_json), FLAGS_write_coco_json_variants,
            FLAGS_write_coco_json_variant, op::String(FLAGS_write_images), op::String(FLAGS_write_images_format),
            op::String(FLAGS_write_video), FLAGS_write_video_fps, FLAGS_write_video_with_audio,
            op::String(FLAGS_write_heatmaps), op::String(FLAGS_write_heatmaps_format), op::String(FLAGS_write_video_3d),
            op::String(FLAGS_write_video_adam), op::String(FLAGS_write_bvh), op::String(FLAGS_udp_host),
            op::String(FLAGS_udp_port)};
G
gineshidalgo99 已提交
155
        opWrapper.configure(wrapperStructOutput);
156
        // No GUI. Equivalent to: opWrapper.configure(op::WrapperStructGui{});
G
gineshidalgo99 已提交
157 158 159
        // Set to single-thread (for sequential processing and/or debugging and/or reducing latency)
        if (FLAGS_disable_multi_thread)
            opWrapper.disableMultiThreading();
G
gineshidalgo99 已提交
160 161 162 163 164 165 166
    }
    catch (const std::exception& e)
    {
        op::error(e.what(), __LINE__, __FUNCTION__, __FILE__);
    }
}

G
gineshidalgo99 已提交
167
int tutorialApiCpp()
G
gineshidalgo99 已提交
168 169 170
{
    try
    {
171
        op::opLog("Starting OpenPose demo...", op::Priority::High);
G
gineshidalgo99 已提交
172
        const auto opTimer = op::getTimerInit();
G
gineshidalgo99 已提交
173

G
gineshidalgo99 已提交
174
        // Required flags to enable heatmaps
175
        FLAGS_body = 0;
G
gineshidalgo99 已提交
176 177 178
        FLAGS_face = true;
        FLAGS_face_detector = 2;

G
gineshidalgo99 已提交
179
        // Configuring OpenPose
180
        op::opLog("Configuring OpenPose...", op::Priority::High);
G
gineshidalgo99 已提交
181 182 183
        op::Wrapper opWrapper{op::ThreadManagerMode::Asynchronous};
        configureWrapper(opWrapper);

G
gineshidalgo99 已提交
184
        // Starting OpenPose
185
        op::opLog("Starting thread(s)...", op::Priority::High);
G
gineshidalgo99 已提交
186
        opWrapper.start();
G
gineshidalgo99 已提交
187

G
gineshidalgo99 已提交
188
        // Read image and face rectangle locations
189 190
        const cv::Mat cvImageToProcess = cv::imread(FLAGS_image_path);
        const op::Matrix imageToProcess = OP_CV2OPCONSTMAT(cvImageToProcess);
G
gineshidalgo99 已提交
191
        const std::vector<op::Rectangle<float>> faceRectangles{
192 193 194
            op::Rectangle<float>{330.119385f, 277.532715f, 48.717274f, 48.717274f}, // Face of person 0
            op::Rectangle<float>{24.036991f, 267.918793f, 65.175171f, 65.175171f},  // Face of person 1
            op::Rectangle<float>{151.803436f, 32.477852f, 108.295761f, 108.295761f} // Face of person 2
G
gineshidalgo99 已提交
195 196 197 198 199 200 201 202 203 204 205 206 207 208
        };

        // Create new datum
        auto datumsPtr = std::make_shared<std::vector<std::shared_ptr<op::Datum>>>();
        datumsPtr->emplace_back();
        auto& datumPtr = datumsPtr->at(0);
        datumPtr = std::make_shared<op::Datum>();
        // Fill datum with image and faceRectangles
        datumPtr->cvInputData = imageToProcess;
        datumPtr->faceRectangles = faceRectangles;

        // Process and display image
        opWrapper.emplaceAndPop(datumsPtr);
        if (datumsPtr != nullptr)
G
gineshidalgo99 已提交
209
        {
G
gineshidalgo99 已提交
210
            printKeypoints(datumsPtr);
G
gineshidalgo99 已提交
211
            if (!FLAGS_no_display)
G
gineshidalgo99 已提交
212
                display(datumsPtr);
G
gineshidalgo99 已提交
213 214
        }
        else
215
            op::opLog("Image could not be processed.", op::Priority::High);
G
gineshidalgo99 已提交
216

G
gineshidalgo99 已提交
217
        // Info
218
        op::opLog("NOTE: In addition with the user flags, this demo has auto-selected the following flags:\n"
G
Gines Hidalgo 已提交
219
                "\t`--body 0 --face --face_detector 2`", op::Priority::High);
G
gineshidalgo99 已提交
220 221 222 223 224

        // Measuring total time
        op::printTime(opTimer, "OpenPose demo successfully finished. Total time: ", " seconds.", op::Priority::High);

        // Return
G
gineshidalgo99 已提交
225 226
        return 0;
    }
G
Gines Hidalgo 已提交
227
    catch (const std::exception&)
G
gineshidalgo99 已提交
228
    {
G
gineshidalgo99 已提交
229
        return -1;
G
gineshidalgo99 已提交
230 231 232 233 234 235 236 237
    }
}

int main(int argc, char *argv[])
{
    // Parsing command line flags
    gflags::ParseCommandLineFlags(&argc, &argv, true);

G
gineshidalgo99 已提交
238 239
    // Running tutorialApiCpp
    return tutorialApiCpp();
G
gineshidalgo99 已提交
240
}