29 #include <opencv2/calib3d.hpp>
30 #include <opencv2/core/eigen.hpp>
31 #include <opencv2/highgui.hpp>
34 #include <SimoxUtility/algorithm/string.h>
47 #include <Calibration/Calibration.h>
48 #include <Calibration/StereoCalibration.h>
49 #include <Image/IplImageAdaptor.h>
67 p.imageProviderName,
"img.ImageProviderName",
"Name of the image provider to use.");
69 p.referenceFrame,
"img.ReferenceFrameName",
"Name of the ReferenceFrameName");
70 defs->optional(p.agentName,
"img.AgentName",
"Name of the agent");
71 defs->optional(p.cameraIndex,
"img.cameraIndex",
"Camera index. Left = 0, Right = 1, ...");
73 std::string pattern =
"[ k3 [, k4, k5, k6 [, s1, s2, s3, s4 ] ] ]";
75 defs->defineOptionalProperty<std::vector<std::string>>(
76 "cam.ExtraDistortionCoeffs",
78 "Optional extra distortion coefficients (which cannot be retrieved from the image "
80 "\nThe expected parameters are: " +
83 "\nThat is, you must specifiy 1 (k3), 4 (k3-k6), or 8 (k3-k6, s1-s4) elements."
84 "\nNote: k1, k2, p1, p2 are retrieved from the image provider.")
87 defs->defineOptionalProperty<
float>(
88 "ar.MarkerSize", p.markerSize.load(),
"The side length of the marker(s)");
89 defs->optional(p.dictionary,
"ar.Dictionary",
"The ArUco dictionary.")
90 .map(
"4X4_50", cv::aruco::DICT_4X4_50)
91 .map(
"4X4_100", cv::aruco::DICT_4X4_100)
92 .map(
"4X4_250", cv::aruco::DICT_4X4_250)
93 .map(
"4X4_1000", cv::aruco::DICT_4X4_1000)
94 .map(
"5X5_50", cv::aruco::DICT_5X5_50)
95 .map(
"5X5_100", cv::aruco::DICT_5X5_100)
96 .map(
"5X5_250", cv::aruco::DICT_5X5_250)
97 .map(
"5X5_1000", cv::aruco::DICT_5X5_1000)
98 .map(
"6X6_50", cv::aruco::DICT_6X6_50)
99 .map(
"6X6_100", cv::aruco::DICT_6X6_100)
100 .map(
"6X6_250", cv::aruco::DICT_6X6_250)
101 .map(
"6X6_1000", cv::aruco::DICT_6X6_1000)
102 .map(
"7X7_50", cv::aruco::DICT_7X7_50)
103 .map(
"7X7_100", cv::aruco::DICT_7X7_100)
104 .map(
"7X7_250", cv::aruco::DICT_7X7_250)
105 .map(
"7X7_1000", cv::aruco::DICT_7X7_1000)
106 .map(
"ARUCO_ORIGINAL", cv::aruco::DICT_ARUCO_ORIGINAL);
108 defs->defineOptionalProperty<std::string>(
"ar.MarkerGeneration",
109 "https://chev.me/arucogen/",
110 "You can generate ArUco markers here.");
112 defs->defineOptionalProperty<
bool>(
113 "visu.Enabled", p.visuEnabled.load(),
"If true, visualize marker poses.");
126 return "ArMarkerLocalizerOpenCV";
133 *arucoDictionary = cv::aruco::getPredefinedDictionary(p.dictionary);
134 p.markerSize = getProperty<float>(
"ar.MarkerSize").getValue();
137 p.visuEnabled = getProperty<bool>(
"visu.Enabled").getValue();
140 std::string propName =
"cam.ExtraDistortionCoeffs";
141 std::vector<std::string> coeffsStr;
142 getProperty(coeffsStr,
"cam.ExtraDistortionCoeffs");
144 p.extraDistortionCoeffs.clear();
145 for (
const auto& coeffStr : coeffsStr)
149 p.extraDistortionCoeffs.push_back(std::stof(coeffStr));
151 catch (
const std::invalid_argument&)
153 ARMARX_WARNING <<
"Could not parse '" << coeffStr <<
"' as float in property "
155 <<
"\nIgnoring extra parameters.";
156 p.extraDistortionCoeffs.clear();
168 robot = virtualRobotReaderPlugin->get().getRobot(p.agentName);
171 referenceNode = robot->getRobotNode(p.referenceFrame);
177 StereoCalibrationInterfacePrx calibrationProvider =
178 StereoCalibrationInterfacePrx::checkedCast(imageProviderInfo.
proxy);
180 std::unique_ptr<CStereoCalibration> stereoCalibration(
182 CCalibration::CCameraParameters ivtCameraParameters =
183 stereoCalibration->GetLeftCalibration()->GetCameraParameters();
192 std::set<int> allowedSizes = {4, 5, 8, 12};
194 <<
" extra distortion coefficients.";
196 int num = int(4 + p.extraDistortionCoeffs.size());
200 simox::alg::multi_to_string(allowedSizes.begin(), allowedSizes.end()),
" ")
201 <<
"\n num = " << num <<
" = 4 + " << p.extraDistortionCoeffs.size()
202 <<
" = 4 + #extra coeffs";
204 cv::Mat distortionParameters(num, 1, cv::DataType<float>::type);
206 if (!calibrationProvider->getImagesAreUndistorted())
209 for (
int i = 0; i < 4; ++i)
211 distortionParameters.at<
float>(i, 0) = ivtCameraParameters.distortion[i];
214 for (
int i = 0; size_t(i) < p.extraDistortionCoeffs.size(); ++i)
216 distortionParameters.at<
float>(4 + i, 0) =
217 p.extraDistortionCoeffs.at(
size_t(i));
222 for (
int i = 0; i < distortionParameters.rows; ++i)
224 distortionParameters.at<
float>(i, 0) = 0;
228 this->distortionCoeffs = distortionParameters;
231 ARMARX_VERBOSE <<
"Distortion coefficients: \n" << this->distortionCoeffs;
234 cameraImages =
new CByteImage*[2];
248 armarx::MetaInfoSizeBasePtr info;
249 getImages(p.imageProviderName, cameraImages, info);
250 m_timestamp_last_image =
261 ArMarkerLocalizationResultList result = localizeAllMarkersInternal();
263 std::unique_lock lock(resultMutex);
264 lastLocalizationResult = result;
272 const Eigen::Isometry3f global_T_frame{referenceNode->getGlobalPose()};
274 for (
const auto& r : result)
276 const Eigen::Isometry3f frame_T_marker{
277 armarx::FramedPosePtr::dynamicCast(r.pose)->toEigen()};
278 const Eigen::Isometry3f global_T_marker = global_T_frame * frame_T_marker;
288 ArMarkerLocalizationResultList
289 ArMarkerLocalizerOpenCV::localizeAllMarkersInternal()
295 cameraImages[0]->bytesPerPixel = 3;
296 cameraImages[1]->bytesPerPixel = 3;
298 IplImage* imageIpl = IplImageAdaptor::Adapt(cameraImages[p.cameraIndex]);
299 cv::Mat imageOpenCV = cv::cvarrToMat(imageIpl);
308 std::vector<int> markerIDs;
309 std::vector<std::vector<cv::Point2f>> markerCorners;
310 cv::aruco::detectMarkers(
311 imageOpenCV, arucoDictionary, markerCorners, markerIDs, arucoParameters);
316 ARMARX_VERBOSE <<
"Localized " << markerIDs.size() <<
" markers: "
317 << simox::alg::join(simox::alg::multi_to_string(markerIDs),
" ");
320 cv::Mat outputImage = imageOpenCV.clone();
321 if (not markerIDs.empty())
323 cv::aruco::drawDetectedMarkers(outputImage, markerCorners, markerIDs);
331 std::vector<cv::Vec3d> rvecs, tvecs;
332 cv::aruco::estimatePoseSingleMarkers(
333 markerCorners, p.markerSize, cameraMatrix, distortionCoeffs, rvecs, tvecs);
336 ARMARX_VERBOSE <<
"Estimated " << rvecs.size() <<
" marker poses.";
340 visionx::ArMarkerLocalizationResultList resultList;
342 for (
size_t i = 0; i < markerIDs.size(); ++i)
344 visionx::ArMarkerLocalizationResult& result = resultList.emplace_back();
347 Eigen::Vector3f position = Eigen::Vector3d(tvec(0), tvec(1), tvec(2)).cast<
float>();
349 cv::Mat cvOrientation;
352 cv::Rodrigues(rvecs.at(i), cvOrientation);
354 catch (
const std::exception& e)
356 ARMARX_ERROR <<
"Caught exception when running cv::aruco::Rodrigues(): \n"
363 Eigen::Matrix3d orientation;
367 orientation.cast<
float>(), position, p.referenceFrame, p.agentName);
368 result.id = markerIDs.at(i);
372 cv::drawFrameAxes(outputImage, cameraMatrix, distortionCoeffs, rvecs[i], tvecs[i], 0.1);
375 CByteImage* output_cimage =
376 new CByteImage(outputImage.size[0], outputImage.size[1], CByteImage::ImageType::eRGB24);
379 CByteImage* result_images[2] = {cameraImages[0], output_cimage};
381 delete output_cimage;
385 visionx::ArMarkerLocalizationResultList
388 if (waitForImages(500))
390 getImages(cameraImages);
391 return localizeAllMarkersInternal();
400 ArMarkerLocalizationResultList
403 std::unique_lock lock(resultMutex);
404 return lastLocalizationResult;
415 tab.markerSize.setRange(1.0, 1000.0);
416 tab.markerSize.setSteps(1000);
417 tab.markerSize.setDecimals(1);
418 tab.markerSize.setValue(p.markerSize);
420 tab.visuEnabled.setValue(p.visuEnabled);
422 grid.
add(
Label(
"Marker size:"), {row, 0}).add(tab.markerSize, {row, 1});
424 grid.
add(
Label(
"Enable visu:"), {row, 0}).add(tab.visuEnabled, {row, 1});
434 if (tab.markerSize.hasValueChanged())
436 p.markerSize = tab.markerSize.getValue();
438 if (tab.visuEnabled.hasValueChanged())
440 p.visuEnabled = tab.visuEnabled.getValue();