forked from minhnh/mas_perception_libs
-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathbounding_box_wrapper.cpp
111 lines (94 loc) · 3.49 KB
/
bounding_box_wrapper.cpp
1
2
3
4
5
6
7
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
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
/*
* Copyright 2018 Bonn-Rhein-Sieg University
*
* Author: Minh Nguyen
*
*/
#include <string>
#include <vector>
#include <utility>
#include <boost/python.hpp>
#include <pcl_conversions/pcl_conversions.h>
#include <geometry_msgs/PoseArray.h>
#include <geometry_msgs/PoseStamped.h>
#include <mas_perception_libs/bounding_box.h>
#include <mas_perception_msgs/BoundingBox.h>
#include <mas_perception_libs/impl/ros_message_serialization.hpp>
#include <mas_perception_libs/bounding_box_wrapper.h>
namespace bp = boost::python;
using BoundingBox = mas_perception_libs::BoundingBox;
using BoundingBoxMsg = mas_perception_msgs::BoundingBox;
namespace mas_perception_libs
{
BoundingBoxWrapper::BoundingBoxWrapper(const bp::object & pSerialCloud, const bp::list &pNormal)
{
sensor_msgs::PointCloud2 rosCloud = from_python<sensor_msgs::PointCloud2>(pSerialCloud);
bp::ssize_t len = bp::len(pNormal);
if (len != 3)
throw std::invalid_argument("expect normal vector to contain 3 numbers of type float");
std::vector<float> normalVect;
for (int i = 0; i < len; ++i)
{
normalVect.push_back(bp::extract<float>(pNormal[i]));
}
const Eigen::Vector3f normal(normalVect[0], normalVect[1], normalVect[2]);
PointCloud::Ptr pclCloudPtr(new PointCloud);
pcl::fromROSMsg(rosCloud, *pclCloudPtr);
mBox = BoundingBox::create(pclCloudPtr->points, normal);
calculatePose(rosCloud.header);
}
BoundingBoxWrapper::~BoundingBoxWrapper() = default;
bp::object BoundingBoxWrapper::getPose()
{
return to_python(mPose);
}
void BoundingBoxWrapper::calculatePose(const std_msgs::Header &header)
{
BoundingBox::Points vertices = mBox.getVertices();
Eigen::Vector3f n1;
Eigen::Vector3f n2;
Eigen::Vector3f n3 = (vertices[4] - vertices[0]) / (vertices[4] - vertices[0]).norm();
// set n1 based on longest edge
if ((vertices[1] - vertices[0]).norm() > (vertices[3] - vertices[0]).norm())
{
n1 = (vertices[1] - vertices[0]) / (vertices[1] - vertices[0]).norm();
}
else
{
n1 = (vertices[3] - vertices[0]) / (vertices[3] - vertices[0]).norm();
}
n2 = n3.cross(n1);
Eigen::Matrix3f m;
m << n1 , n2 , n3;
Eigen::Quaternion<float> q(m);
q.normalize();
Eigen::Vector3f centroid = mBox.getCenter();
mPose.header = header;
mPose.pose.position.x = centroid(0);
mPose.pose.position.y = centroid(1);
mPose.pose.position.z = centroid(2);
mPose.pose.orientation.x = q.x();
mPose.pose.orientation.y = q.y();
mPose.pose.orientation.z = q.z();
mPose.pose.orientation.w = q.w();
}
bp::object BoundingBoxWrapper::getRosMsg()
{
BoundingBoxMsg boxMsg;
boxMsg.center = mPose.pose.position;
Eigen::Vector3f dimensions = mBox.getDimensions();
boxMsg.dimensions.x = dimensions(0);
boxMsg.dimensions.y = dimensions(1);
boxMsg.dimensions.z = dimensions(2);
BoundingBox::Points vertices = mBox.getVertices();
for (size_t i = 0; i < vertices.size(); ++i)
{
geometry_msgs::Point vertice;
vertice.x = vertices[i](0);
vertice.y = vertices[i](1);
vertice.z = vertices[i](2);
boxMsg.vertices.push_back(vertice);
}
return to_python(boxMsg);
}
} // namespace mas_perception_libs