07_hand_from_image.cpp 11.5 KB
Newer Older
1
// ----------------------------- OpenPose C++ API Tutorial - Example 7 - Face from Image -----------------------------
G
gineshidalgo99 已提交
2 3 4 5
// It reads an image and the hand location, process it, and displays the hand keypoints. In addition,
// it includes all the OpenPose configuration flags.
// Input: An image and the hand rectangle locations.
// Output: OpenPose hand keypoint detection.
G
Gines Hidalgo 已提交
6
// NOTE: This demo is auto-selecting the following flags: `--body 0 --hand --hand_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 115
        // Face and hand detectors
        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 133
            FLAGS_face, faceDetector, faceNetInputSize,
            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_hand = true;
        FLAGS_hand_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 hand 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<std::array<op::Rectangle<float>, 2>> handRectangles{
192
            // Left/Right hands of person 0
G
gineshidalgo99 已提交
193
            std::array<op::Rectangle<float>, 2>{
194 195 196
                op::Rectangle<float>{320.035889f, 377.675049f, 69.300949f, 69.300949f}, // Left hand
                op::Rectangle<float>{0.f, 0.f, 0.f, 0.f}},                              // Right hand
            // Left/Right hands of person 1
G
gineshidalgo99 已提交
197
            std::array<op::Rectangle<float>, 2>{
198 199 200
                op::Rectangle<float>{80.155792f, 407.673492f, 80.812706f, 80.812706f},  // Left hand
                op::Rectangle<float>{46.449715f, 404.559753f, 98.898178f, 98.898178f}}, // Right hand
            // Left/Right hands of person 2
G
gineshidalgo99 已提交
201
            std::array<op::Rectangle<float>, 2>{
202 203
                op::Rectangle<float>{185.692673f, 303.112244f, 157.587555f, 157.587555f},// Left hand
                op::Rectangle<float>{88.984360f, 268.866547f, 117.818230f, 117.818230f}} // Right hand
G
gineshidalgo99 已提交
204 205 206 207 208 209 210 211 212 213 214 215 216 217
        };

        // 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 handRectangles
        datumPtr->cvInputData = imageToProcess;
        datumPtr->handRectangles = handRectangles;

        // Process and display image
        opWrapper.emplaceAndPop(datumsPtr);
        if (datumsPtr != nullptr)
G
gineshidalgo99 已提交
218
        {
G
gineshidalgo99 已提交
219
            printKeypoints(datumsPtr);
G
gineshidalgo99 已提交
220
            if (!FLAGS_no_display)
G
gineshidalgo99 已提交
221
                display(datumsPtr);
G
gineshidalgo99 已提交
222 223
        }
        else
224
            op::opLog("Image could not be processed.", op::Priority::High);
G
gineshidalgo99 已提交
225

G
gineshidalgo99 已提交
226
        // Info
227
        op::opLog("NOTE: In addition with the user flags, this demo has auto-selected the following flags:\n"
G
Gines Hidalgo 已提交
228
                "\t`--body 0 --hand --hand_detector 2`", op::Priority::High);
G
gineshidalgo99 已提交
229 230 231 232 233

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

        // Return
G
gineshidalgo99 已提交
234 235
        return 0;
    }
G
Gines Hidalgo 已提交
236
    catch (const std::exception&)
G
gineshidalgo99 已提交
237
    {
G
gineshidalgo99 已提交
238
        return -1;
G
gineshidalgo99 已提交
239 240 241 242 243 244 245 246
    }
}

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

G
gineshidalgo99 已提交
247 248
    // Running tutorialApiCpp
    return tutorialApiCpp();
G
gineshidalgo99 已提交
249
}