22 #include <Eigen/Eigen> 23 #include <boost/algorithm/string/predicate.hpp> 24 #include <boost/filesystem.hpp> 25 #include <boost/foreach.hpp> 37 #ifdef HAVE_PYTHONLIBS 45 void plot_xy_positions(
const std::string &name,
const std::string &color,
const std::vector<Eigen::Matrix<double, 7, 1>> &poses) {
48 std::map<std::string, std::string> params;
49 params.insert({
"label", name});
50 params.insert({
"linestyle",
"-"});
51 params.insert({
"color", color});
54 std::vector<double> x, y;
55 for (
size_t i = 0; i < poses.size(); i++) {
56 x.push_back(poses.at(i)(0));
57 y.push_back(poses.at(i)(1));
65 void plot_z_positions(
const std::string &name,
const std::string &color,
const std::vector<double> ×,
66 const std::vector<Eigen::Matrix<double, 7, 1>> &poses) {
69 std::map<std::string, std::string> params;
70 params.insert({
"label", name});
71 params.insert({
"linestyle",
"-"});
72 params.insert({
"color", color});
75 std::vector<double> time, z;
76 for (
size_t i = 0; i < poses.size(); i++) {
77 time.push_back(times.at(i));
78 z.push_back(poses.at(i)(2));
87 int main(
int argc,
char **argv) {
95 PRINT_ERROR(
RED "ERROR: ./plot_trajectories <align_mode> <file_gt.txt> <file_est1.txt> ... <file_est9.txt>\n" RESET);
96 PRINT_ERROR(
RED "ERROR: rosrun ov_eval plot_trajectories <align_mode> <file_gt.txt> <file_est1.txt> ... <file_est9.txt>\n" RESET);
97 std::exit(EXIT_FAILURE);
101 std::vector<std::string> names;
102 std::vector<std::vector<double>> times;
103 std::vector<std::vector<Eigen::Matrix<double, 7, 1>>> poses;
104 for (
int i = 2; i < argc; i++) {
107 std::vector<double> times_temp;
108 std::vector<Eigen::Matrix<double, 7, 1>> poses_temp;
109 std::vector<Eigen::Matrix3d> cov_ori_temp, cov_pos_temp;
116 std::vector<double> gt_times_temp(times.at(0));
117 std::vector<Eigen::Matrix<double, 7, 1>> gt_poses_temp(poses.at(0));
121 if (poses_temp.size() < 3) {
122 PRINT_ERROR(
RED "[TRAJ]: unable to get enough common timestamps between trajectories.\n" RESET);
123 PRINT_ERROR(
RED "[TRAJ]: does the estimated trajectory publish the rosbag timestamps??\n" RESET);
124 std::exit(EXIT_FAILURE);
128 Eigen::Matrix3d R_ESTtoGT;
129 Eigen::Vector3d t_ESTinGT;
135 PRINT_DEBUG(
"[TRAJ]: q_ESTtoGT = %.3f, %.3f, %.3f, %.3f | p_ESTinGT = %.3f, %.3f, %.3f | s = %.2f\n", q_ESTtoGT(0), q_ESTtoGT(1),
136 q_ESTtoGT(2), q_ESTtoGT(3), t_ESTinGT(0), t_ESTinGT(1), t_ESTinGT(2), s_ESTtoGT);
139 std::vector<Eigen::Matrix<double, 7, 1>> est_poses_aignedtoGT;
140 for (
size_t j = 0; j < gt_times_temp.size(); j++) {
141 Eigen::Matrix<double, 7, 1> pose_ESTinGT;
142 pose_ESTinGT.block(0, 0, 3, 1) = s_ESTtoGT * R_ESTtoGT * poses_temp.at(j).block(0, 0, 3, 1) + t_ESTinGT;
144 est_poses_aignedtoGT.push_back(pose_ESTinGT);
148 poses_temp = est_poses_aignedtoGT;
152 boost::filesystem::path path(argv[i]);
153 std::string name = path.stem().string();
155 PRINT_INFO(
"[COMP]: %d poses in %s => length of %.2f meters\n", (
int)times_temp.size(), name.c_str(), length);
158 names.push_back(name);
159 times.push_back(times_temp);
160 poses.push_back(poses_temp);
163 #ifdef HAVE_PYTHONLIBS 166 std::vector<std::string> colors = {
"black",
"blue",
"red",
"green",
"cyan",
"magenta"};
173 for (
size_t i = 0; i < times.size(); i++) {
174 plot_xy_positions(names.at(i), colors.at(i), poses.at(i));
187 double starttime = (times.at(0).empty()) ? 0 : times.at(0).at(0);
188 double endtime = (times.at(0).empty()) ? 0 : times.at(0).at(times.at(0).size() - 1);
189 for (
size_t i = 0; i < times.size(); i++) {
190 for (
size_t j = 0; j < times.at(i).size(); j++) {
191 times.at(i).at(j) -= starttime;
196 for (
size_t i = 0; i < times.size(); i++) {
197 plot_z_positions(names.at(i), colors.at(i), times.at(i), poses.at(i));
Eigen::Matrix< double, 4, 1 > quat_multiply(const Eigen::Matrix< double, 4, 1 > &q, const Eigen::Matrix< double, 4, 1 > &p)
static void load_data(std::string path_traj, std::vector< double > ×, std::vector< Eigen::Matrix< double, 7, 1 >> &poses, std::vector< Eigen::Matrix3d > &cov_ori, std::vector< Eigen::Matrix3d > &cov_pos)
This will load space separated trajectory into memory.
static void perform_association(double offset, double max_difference, std::vector< double > &est_times, std::vector< double > >_times, std::vector< Eigen::Matrix< double, 7, 1 >> &est_poses, std::vector< Eigen::Matrix< double, 7, 1 >> >_poses)
Will intersect our loaded data so that we have common timestamps.
Eigen::Matrix< double, 4, 1 > Inv(Eigen::Matrix< double, 4, 1 > q)
void figure_size(size_t w, size_t h)
bool plot(const std::vector< Numeric > &x, const std::vector< Numeric > &y, const std::map< std::string, std::string > &keywords)
static double get_total_length(const std::vector< Eigen::Matrix< double, 7, 1 >> &poses)
Will calculate the total trajectory distance.
static void align_trajectory(const std::vector< Eigen::Matrix< double, 7, 1 >> &traj_es, const std::vector< Eigen::Matrix< double, 7, 1 >> &traj_gt, Eigen::Matrix3d &R, Eigen::Vector3d &t, double &s, std::string method, int n_aligned=-1)
Align estimate to GT using a desired method using a set of initial poses.
void show(const bool block=true)
int main(int argc, char **argv)
Eigen::Matrix< double, 4, 1 > rot_2_quat(const Eigen::Matrix< double, 3, 3 > &rot)
static void setPrintLevel(const std::string &level)
#define PRINT_ERROR(x...)
void xlim(Numeric left, Numeric right)
void xlabel(const std::string &str, const std::map< std::string, std::string > &keywords={})
void ylabel(const std::string &str, const std::map< std::string, std::string > &keywords={})
#define PRINT_DEBUG(x...)