08_heatmaps_from_image.cpp 12.1 KB
Newer Older
G
gineshidalgo99 已提交
1 2 3 4
// ----------------------- OpenPose C++ API Tutorial - Example 8 - Heatmaps from image -----------------------
// It reads an image, process it, and displays it with the body heatmaps. In addition, it includes all the
// OpenPose configuration flags (enable/disable hand, face, output saving, etc.).

5 6 7
// Third-party dependencies
#include <opencv2/opencv.hpp>
// Command-line user interface
G
gineshidalgo99 已提交
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
#define OPENPOSE_FLAGS_DISABLE_PRODUCER
#define OPENPOSE_FLAGS_DISABLE_DISPLAY
#include <openpose/flags.hpp>
// OpenPose dependencies
#include <openpose/headers.hpp>

// Custom OpenPose flags
// Producer
DEFINE_string(image_path, "examples/media/COCO_val2014_000000000294.jpg",
    "Process an image. Read all standard formats (jpg, png, bmp, etc.).");
// Display
DEFINE_bool(no_display,                 false,
    "Enable to disable the visual display.");

// This worker will just read and return all the jpg files in a directory
bool display(
    const std::shared_ptr<std::vector<std::shared_ptr<op::Datum>>>& datumsPtr, const int desiredChannel = 0)
{
    try
    {
        if (datumsPtr != nullptr && !datumsPtr->empty())
        {
            // Note: Heatmaps are in net_resolution size, which does not necessarily match the final image size
            // Read heatmaps
            auto& poseHeatMaps = datumsPtr->at(0)->poseHeatMaps;
            // Read desired channel
            const auto numberChannels = poseHeatMaps.getSize(0);
            const auto height = poseHeatMaps.getSize(1);
            const auto width = poseHeatMaps.getSize(2);
            const cv::Mat desiredChannelHeatMap(
                height, width, CV_32F, &poseHeatMaps.getPtr()[desiredChannel % numberChannels*height*width]);
            // Read image used from OpenPose body network (same resolution than heatmaps)
            auto& inputNetData = datumsPtr->at(0)->inputNetData[0];
            const cv::Mat inputNetDataB(
                height, width, CV_32F, &inputNetData.getPtr()[0]);
            const cv::Mat inputNetDataG(
                height, width, CV_32F, &inputNetData.getPtr()[height*width]);
            const cv::Mat inputNetDataR(
                height, width, CV_32F, &inputNetData.getPtr()[2*height*width]);
            cv::Mat netInputImage;
R
Raaj 已提交
48
            cv::merge(std::vector<cv::Mat>{inputNetDataB, inputNetDataG, inputNetDataR}, netInputImage);
G
gineshidalgo99 已提交
49 50 51 52 53 54 55 56 57 58 59
            netInputImage = (netInputImage+0.5)*255;
            // Turn into uint8 cv::Mat
            cv::Mat netInputImageUint8;
            netInputImage.convertTo(netInputImageUint8, CV_8UC1);
            cv::Mat desiredChannelHeatMapUint8;
            desiredChannelHeatMap.convertTo(desiredChannelHeatMapUint8, CV_8UC1);
            // Combining both images
            cv::Mat imageToRender;
            cv::applyColorMap(desiredChannelHeatMapUint8, desiredChannelHeatMapUint8, cv::COLORMAP_JET);
            cv::addWeighted(netInputImageUint8, 0.5, desiredChannelHeatMapUint8, 0.5, 0., imageToRender);
            // Display image
60 61 62 63
            if (!imageToRender.empty())
                cv::imshow(OPEN_POSE_NAME_AND_VERSION + " - Tutorial C++ API", imageToRender);
            else
                op::opLog("Empty cv::Mat as output.", op::Priority::High, __LINE__, __FUNCTION__, __FILE__);
G
gineshidalgo99 已提交
64 65
        }
        else
66
            op::opLog("Nullptr or empty datumsPtr found.", op::Priority::High);
G
gineshidalgo99 已提交
67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87
        const auto key = (char)cv::waitKey(1);
        return (key == 27);
    }
    catch (const std::exception& e)
    {
        op::error(e.what(), __LINE__, __FUNCTION__, __FILE__);
        return true;
    }
}

void printKeypoints(const std::shared_ptr<std::vector<std::shared_ptr<op::Datum>>>& datumsPtr)
{
    try
    {
        // Example: How to use the pose keypoints
        if (datumsPtr != nullptr && !datumsPtr->empty())
        {
            const auto& poseHeatMaps = datumsPtr->at(0)->poseHeatMaps;
            const auto numberChannels = poseHeatMaps.getSize(0);
            const auto height = poseHeatMaps.getSize(1);
            const auto width = poseHeatMaps.getSize(2);
88
            op::opLog("Body heatmaps has " + std::to_string(numberChannels) + " channels, and each channel has a"
G
gineshidalgo99 已提交
89 90 91 92
                    " dimension of " + std::to_string(width) + " x " + std::to_string(height) + " pixels.",
                    op::Priority::High);
        }
        else
93
            op::opLog("Nullptr or empty datumsPtr found.", op::Priority::High);
G
gineshidalgo99 已提交
94 95 96 97 98 99 100 101 102 103 104 105 106 107
    }
    catch (const std::exception& e)
    {
        op::error(e.what(), __LINE__, __FUNCTION__, __FILE__);
    }
}

void configureWrapper(op::Wrapper& opWrapper)
{
    try
    {
        // Configuring OpenPose

        // logging_level
108 109 110
        op::checkBool(
            0 <= FLAGS_logging_level && FLAGS_logging_level <= 255, "Wrong logging_level value.",
            __LINE__, __FUNCTION__, __FILE__);
G
gineshidalgo99 已提交
111 112 113 114 115
        op::ConfigureLog::setPriorityThreshold((op::Priority)FLAGS_logging_level);
        op::Profiler::setDefaultX(FLAGS_profile_speed);

        // Applying user defined configuration - GFlags to program variables
        // outputSize
G
Gines Hidalgo 已提交
116
        const auto outputSize = op::flagsToPoint(op::String(FLAGS_output_resolution), "-1x-1");
G
gineshidalgo99 已提交
117
        // netInputSize
G
Gines Hidalgo 已提交
118
        const auto netInputSize = op::flagsToPoint(op::String(FLAGS_net_resolution), "-1x368");
G
gineshidalgo99 已提交
119
        // faceNetInputSize
G
Gines Hidalgo 已提交
120
        const auto faceNetInputSize = op::flagsToPoint(op::String(FLAGS_face_net_resolution), "368x368 (multiples of 16)");
G
gineshidalgo99 已提交
121
        // handNetInputSize
G
Gines Hidalgo 已提交
122
        const auto handNetInputSize = op::flagsToPoint(op::String(FLAGS_hand_net_resolution), "368x368 (multiples of 16)");
123 124
        // poseMode
        const auto poseMode = op::flagsToPoseMode(FLAGS_body);
G
gineshidalgo99 已提交
125
        // poseModel
G
Gines Hidalgo 已提交
126
        const auto poseModel = op::flagsToPoseModel(op::String(FLAGS_model_pose));
G
gineshidalgo99 已提交
127 128
        // JSON saving
        if (!FLAGS_write_keypoint.empty())
129 130 131
            op::opLog(
                "Flag `write_keypoint` is deprecated and will eventually be removed. Please, use `write_json`"
                " instead.", op::Priority::Max);
G
gineshidalgo99 已提交
132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
        // keypointScaleMode
        const auto keypointScaleMode = op::flagsToScaleMode(FLAGS_keypoint_scale);
        // heatmaps to add
        const auto heatMapTypes = op::flagsToHeatMaps(FLAGS_heatmaps_add_parts, FLAGS_heatmaps_add_bkg,
                                                      FLAGS_heatmaps_add_PAFs);
        const auto heatMapScaleMode = op::flagsToHeatMapScaleMode(FLAGS_heatmaps_scale);
        // >1 camera view?
        const auto multipleView = (FLAGS_3d || FLAGS_3d_views > 1);
        // Face and hand detectors
        const auto faceDetector = op::flagsToDetector(FLAGS_face_detector);
        const auto handDetector = op::flagsToDetector(FLAGS_hand_detector);
        // Enabling Google Logging
        const bool enableGoogleLogging = true;

        // Pose configuration (use WrapperStructPose{} for default and recommended configuration)
        const op::WrapperStructPose wrapperStructPose{
148
            poseMode, netInputSize, outputSize, keypointScaleMode, FLAGS_num_gpu, FLAGS_num_gpu_start,
G
gineshidalgo99 已提交
149 150
            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 已提交
151
            FLAGS_part_to_show, op::String(FLAGS_model_folder), heatMapTypes, heatMapScaleMode, FLAGS_part_candidates,
G
gineshidalgo99 已提交
152
            (float)FLAGS_render_threshold, FLAGS_number_people_max, FLAGS_maximize_positives, FLAGS_fps_max,
G
Gines Hidalgo 已提交
153 154
            op::String(FLAGS_prototxt_path), op::String(FLAGS_caffemodel_path),
            (float)FLAGS_upsampling_ratio, enableGoogleLogging};
G
gineshidalgo99 已提交
155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173
        opWrapper.configure(wrapperStructPose);
        // Face configuration (use op::WrapperStructFace{} to disable it)
        const op::WrapperStructFace wrapperStructFace{
            FLAGS_face, faceDetector, faceNetInputSize,
            op::flagsToRenderMode(FLAGS_face_render, multipleView, FLAGS_render_pose),
            (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{
            FLAGS_hand, handDetector, handNetInputSize, FLAGS_hand_scale_number, (float)FLAGS_hand_scale_range,
            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);
        // Output (comment or use default argument to disable any output)
        const op::WrapperStructOutput wrapperStructOutput{
G
Gines Hidalgo 已提交
174 175 176 177 178 179 180
            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 已提交
181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196
        opWrapper.configure(wrapperStructOutput);
        // No GUI. Equivalent to: opWrapper.configure(op::WrapperStructGui{});
        // Set to single-thread (for sequential processing and/or debugging and/or reducing latency)
        if (FLAGS_disable_multi_thread)
            opWrapper.disableMultiThreading();
    }
    catch (const std::exception& e)
    {
        op::error(e.what(), __LINE__, __FUNCTION__, __FILE__);
    }
}

int tutorialApiCpp()
{
    try
    {
197
        op::opLog("Starting OpenPose demo...", op::Priority::High);
G
gineshidalgo99 已提交
198 199 200 201 202 203 204 205 206
        const auto opTimer = op::getTimerInit();

        // Required flags to enable heatmaps
        FLAGS_heatmaps_add_parts = true;
        FLAGS_heatmaps_add_bkg = true;
        FLAGS_heatmaps_add_PAFs = true;
        FLAGS_heatmaps_scale = 2;

        // Configuring OpenPose
207
        op::opLog("Configuring OpenPose...", op::Priority::High);
G
gineshidalgo99 已提交
208 209 210 211
        op::Wrapper opWrapper{op::ThreadManagerMode::Asynchronous};
        configureWrapper(opWrapper);

        // Starting OpenPose
212
        op::opLog("Starting thread(s)...", op::Priority::High);
G
gineshidalgo99 已提交
213 214 215
        opWrapper.start();

        // Process and display image
216 217
        const cv::Mat cvImageToProcess = cv::imread(FLAGS_image_path);
        const op::Matrix imageToProcess = OP_CV2OPCONSTMAT(cvImageToProcess);
G
gineshidalgo99 已提交
218 219 220 221 222 223 224 225 226 227 228 229 230
        auto datumProcessed = opWrapper.emplaceAndPop(imageToProcess);
        if (datumProcessed != nullptr)
        {
            printKeypoints(datumProcessed);
            if (!FLAGS_no_display)
            {
                const auto numberChannels = datumProcessed->at(0)->poseHeatMaps.getSize(0);
                for (auto desiredChannel = 0 ; desiredChannel < numberChannels ; desiredChannel++)
                    if (display(datumProcessed, desiredChannel))
                        break;
            }
        }
        else
231
            op::opLog("Image could not be processed.", op::Priority::High);
G
gineshidalgo99 已提交
232 233

        // Info
234
        op::opLog("NOTE: In addition with the user flags, this demo has auto-selected the following flags:\n"
235
                "\t`--heatmaps_add_parts --heatmaps_add_bkg --heatmaps_add_PAFs`",
G
gineshidalgo99 已提交
236 237 238 239 240 241 242 243
                op::Priority::High);

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

        // Return
        return 0;
    }
G
Gines Hidalgo 已提交
244
    catch (const std::exception&)
G
gineshidalgo99 已提交
245 246 247 248 249 250 251 252 253 254 255 256 257
    {
        return -1;
    }
}

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

    // Running tutorialApiCpp
    return tutorialApiCpp();
}