orb_slam3_details/src/FrameDrawer.cc

407 lines
12 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

/**
* This file is part of ORB-SLAM3
*
* Copyright (C) 2017-2020 Carlos Campos, Richard Elvira, Juan J. Gómez Rodríguez, José M.M. Montiel and Juan D. Tardós, University of Zaragoza.
* Copyright (C) 2014-2016 Raúl Mur-Artal, José M.M. Montiel and Juan D. Tardós, University of Zaragoza.
*
* ORB-SLAM3 is free software: you can redistribute it and/or modify it under the terms of the GNU General Public
* License as published by the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* ORB-SLAM3 is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with ORB-SLAM3.
* If not, see <http://www.gnu.org/licenses/>.
*/
#include "FrameDrawer.h"
#include "Tracking.h"
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
#include<mutex>
namespace ORB_SLAM3
{
FrameDrawer::FrameDrawer(Atlas* pAtlas):both(false),mpAtlas(pAtlas)
{
mState=Tracking::SYSTEM_NOT_READY;
// 初始化图像显示画布
// 包括图像、特征点连线形成的轨迹初始化时、框跟踪时的MapPoint、圈跟踪时的特征点
mIm = cv::Mat(480,640,CV_8UC3, cv::Scalar(0,0,0));
mImRight = cv::Mat(480,640,CV_8UC3, cv::Scalar(0,0,0));
}
cv::Mat FrameDrawer::DrawFrame(bool bOldFeatures)
{
cv::Mat im;
vector<cv::KeyPoint> vIniKeys; // Initialization: KeyPoints in reference frame
vector<int> vMatches; // Initialization: correspondeces with reference keypoints
vector<cv::KeyPoint> vCurrentKeys; // KeyPoints in current frame
vector<bool> vbVO, vbMap; // Tracked MapPoints in current frame
vector<pair<cv::Point2f, cv::Point2f> > vTracks;
int state; // Tracking state
Frame currentFrame;
vector<MapPoint*> vpLocalMap;
vector<cv::KeyPoint> vMatchesKeys;
vector<MapPoint*> vpMatchedMPs;
vector<cv::KeyPoint> vOutlierKeys;
vector<MapPoint*> vpOutlierMPs;
map<long unsigned int, cv::Point2f> mProjectPoints;
map<long unsigned int, cv::Point2f> mMatchedInImage;
//Copy variables within scoped mutex
{
unique_lock<mutex> lock(mMutex);
state=mState;
if(mState==Tracking::SYSTEM_NOT_READY)
mState=Tracking::NO_IMAGES_YET;
mIm.copyTo(im);
if(mState==Tracking::NOT_INITIALIZED)
{
vCurrentKeys = mvCurrentKeys;
vIniKeys = mvIniKeys;
vMatches = mvIniMatches;
vTracks = mvTracks;
}
else if(mState==Tracking::OK)
{
vCurrentKeys = mvCurrentKeys;
vbVO = mvbVO;
vbMap = mvbMap;
currentFrame = mCurrentFrame;
vpLocalMap = mvpLocalMap;
vMatchesKeys = mvMatchedKeys;
vpMatchedMPs = mvpMatchedMPs;
vOutlierKeys = mvOutlierKeys;
vpOutlierMPs = mvpOutlierMPs;
mProjectPoints = mmProjectPoints;
mMatchedInImage = mmMatchedInImage;
}
else if(mState==Tracking::LOST)
{
vCurrentKeys = mvCurrentKeys;
}
}
if(im.channels()<3) //this should be always true
cvtColor(im,im,cv::COLOR_GRAY2BGR);
//Draw
if(state==Tracking::NOT_INITIALIZED)
{
for(unsigned int i=0; i<vMatches.size(); i++)
{
if(vMatches[i]>=0)
{
cv::line(im,vIniKeys[i].pt,vCurrentKeys[vMatches[i]].pt,
cv::Scalar(0,255,0));
}
}
for(vector<pair<cv::Point2f, cv::Point2f> >::iterator it=vTracks.begin(); it!=vTracks.end(); it++)
cv::line(im,(*it).first,(*it).second, cv::Scalar(0,255,0),5);
}
else if(state==Tracking::OK && bOldFeatures) //TRACKING
{
mnTracked=0;
mnTrackedVO=0;
const float r = 5;
int n = vCurrentKeys.size();
for(int i=0;i<n;i++)
{
if(vbVO[i] || vbMap[i])
{
cv::Point2f pt1,pt2;
pt1.x=vCurrentKeys[i].pt.x-r;
pt1.y=vCurrentKeys[i].pt.y-r;
pt2.x=vCurrentKeys[i].pt.x+r;
pt2.y=vCurrentKeys[i].pt.y+r;
// This is a match to a MapPoint in the map
if(vbMap[i])
{
cv::rectangle(im,pt1,pt2,cv::Scalar(0,255,0));
cv::circle(im,vCurrentKeys[i].pt,2,cv::Scalar(0,255,0),-1);
mnTracked++;
}
else // This is match to a "visual odometry" MapPoint created in the last frame
{
cv::rectangle(im,pt1,pt2,cv::Scalar(255,0,0));
cv::circle(im,vCurrentKeys[i].pt,2,cv::Scalar(255,0,0),-1);
mnTrackedVO++;
}
}
}
}
else if(state==Tracking::OK && !bOldFeatures)
{
mnTracked=0;
int nTracked2 = 0;
mnTrackedVO=0;
int n = vCurrentKeys.size();
for(int i=0; i < n; ++i)
{
// This is a match to a MapPoint in the map
if(vbMap[i])
{
mnTracked++;
}
}
map<long unsigned int, cv::Point2f>::iterator it_match = mMatchedInImage.begin();
while(it_match != mMatchedInImage.end())
{
long unsigned int mp_id = it_match->first;
cv::Point2f p_image = it_match->second;
if(mProjectPoints.find(mp_id) != mProjectPoints.end())
{
cv::Point2f p_proj = mMatchedInImage[mp_id];
cv::line(im, p_proj, p_image, cv::Scalar(0, 255, 0), 2);
nTracked2++;
}
else
{
cv::circle(im,p_image,2,cv::Scalar(0,0,255),-1);
}
it_match++;
}
n = vOutlierKeys.size();
for(int i=0; i < n; ++i)
{
cv::Point2f point3d_proy;
float u, v;
currentFrame.ProjectPointDistort(vpOutlierMPs[i] , point3d_proy, u, v);
cv::Point2f point_im = vOutlierKeys[i].pt;
cv::line(im,cv::Point2f(u, v), point_im,cv::Scalar(0, 0, 255), 1);
}
}
cv::Mat imWithInfo;
DrawTextInfo(im,state, imWithInfo);
return imWithInfo;
}
cv::Mat FrameDrawer::DrawRightFrame()
{
cv::Mat im;
vector<cv::KeyPoint> vIniKeys; // Initialization: KeyPoints in reference frame
vector<int> vMatches; // Initialization: correspondeces with reference keypoints
vector<cv::KeyPoint> vCurrentKeys; // KeyPoints in current frame
vector<bool> vbVO, vbMap; // Tracked MapPoints in current frame
int state; // Tracking state
//Copy variables within scoped mutex
{
unique_lock<mutex> lock(mMutex);
state=mState;
if(mState==Tracking::SYSTEM_NOT_READY)
mState=Tracking::NO_IMAGES_YET;
mImRight.copyTo(im);
if(mState==Tracking::NOT_INITIALIZED)
{
vCurrentKeys = mvCurrentKeysRight;
vIniKeys = mvIniKeys;
vMatches = mvIniMatches;
}
else if(mState==Tracking::OK)
{
vCurrentKeys = mvCurrentKeysRight;
vbVO = mvbVO;
vbMap = mvbMap;
}
else if(mState==Tracking::LOST)
{
vCurrentKeys = mvCurrentKeysRight;
}
} // destroy scoped mutex -> release mutex
if(im.channels()<3) //this should be always true
cvtColor(im,im,cv::COLOR_GRAY2BGR);
//Draw
if(state==Tracking::NOT_INITIALIZED) //INITIALIZING
{
for(unsigned int i=0; i<vMatches.size(); i++)
{
if(vMatches[i]>=0)
{
cv::line(im,vIniKeys[i].pt,vCurrentKeys[vMatches[i]].pt,
cv::Scalar(0,255,0));
}
}
}
else if(state==Tracking::OK) //TRACKING
{
mnTracked=0;
mnTrackedVO=0;
const float r = 5;
const int n = mvCurrentKeysRight.size();
const int Nleft = mvCurrentKeys.size();
for(int i=0;i<n;i++)
{
if(vbVO[i + Nleft] || vbMap[i + Nleft])
{
cv::Point2f pt1,pt2;
pt1.x=mvCurrentKeysRight[i].pt.x-r;
pt1.y=mvCurrentKeysRight[i].pt.y-r;
pt2.x=mvCurrentKeysRight[i].pt.x+r;
pt2.y=mvCurrentKeysRight[i].pt.y+r;
// This is a match to a MapPoint in the map
if(vbMap[i + Nleft])
{
cv::rectangle(im,pt1,pt2,cv::Scalar(0,255,0));
cv::circle(im,mvCurrentKeysRight[i].pt,2,cv::Scalar(0,255,0),-1);
mnTracked++;
}
else // This is match to a "visual odometry" MapPoint created in the last frame
{
cv::rectangle(im,pt1,pt2,cv::Scalar(255,0,0));
cv::circle(im,mvCurrentKeysRight[i].pt,2,cv::Scalar(255,0,0),-1);
mnTrackedVO++;
}
}
}
}
cv::Mat imWithInfo;
DrawTextInfo(im,state, imWithInfo);
return imWithInfo;
}
void FrameDrawer::DrawTextInfo(cv::Mat &im, int nState, cv::Mat &imText)
{
stringstream s;
if(nState==Tracking::NO_IMAGES_YET)
s << " WAITING FOR IMAGES";
else if(nState==Tracking::NOT_INITIALIZED)
s << " TRYING TO INITIALIZE ";
else if(nState==Tracking::OK)
{
if(!mbOnlyTracking)
s << "SLAM MODE | ";
else
s << "LOCALIZATION | ";
int nMaps = mpAtlas->CountMaps();
int nKFs = mpAtlas->KeyFramesInMap();
int nMPs = mpAtlas->MapPointsInMap();
s << "Maps: " << nMaps << ", KFs: " << nKFs << ", MPs: " << nMPs << ", Matches: " << mnTracked;
if(mnTrackedVO>0)
s << ", + VO matches: " << mnTrackedVO;
}
else if(nState==Tracking::LOST)
{
s << " TRACK LOST. TRYING TO RELOCALIZE ";
}
else if(nState==Tracking::SYSTEM_NOT_READY)
{
s << " LOADING ORB VOCABULARY. PLEASE WAIT...";
}
int baseline=0;
cv::Size textSize = cv::getTextSize(s.str(),cv::FONT_HERSHEY_PLAIN,1,1,&baseline);
imText = cv::Mat(im.rows+textSize.height+10,im.cols,im.type());
im.copyTo(imText.rowRange(0,im.rows).colRange(0,im.cols));
imText.rowRange(im.rows,imText.rows) = cv::Mat::zeros(textSize.height+10,im.cols,im.type());
cv::putText(imText,s.str(),cv::Point(5,imText.rows-5),cv::FONT_HERSHEY_PLAIN,1,cv::Scalar(255,255,255),1,8);
}
void FrameDrawer::Update(Tracking *pTracker)
{
unique_lock<mutex> lock(mMutex);
pTracker->mImGray.copyTo(mIm);
mvCurrentKeys=pTracker->mCurrentFrame.mvKeys;
if(both){
mvCurrentKeysRight = pTracker->mCurrentFrame.mvKeysRight;
pTracker->mImRight.copyTo(mImRight);
N = mvCurrentKeys.size() + mvCurrentKeysRight.size();
}
else{
N = mvCurrentKeys.size();
}
mvbVO = vector<bool>(N,false);
mvbMap = vector<bool>(N,false);
mbOnlyTracking = pTracker->mbOnlyTracking;
//Variables for the new visualization
mCurrentFrame = pTracker->mCurrentFrame;
mmProjectPoints = mCurrentFrame.mmProjectPoints;
mmMatchedInImage.clear();
mvpLocalMap = pTracker->GetLocalMapMPS();
mvMatchedKeys.clear();
mvMatchedKeys.reserve(N);
mvpMatchedMPs.clear();
mvpMatchedMPs.reserve(N);
mvOutlierKeys.clear();
mvOutlierKeys.reserve(N);
mvpOutlierMPs.clear();
mvpOutlierMPs.reserve(N);
if(pTracker->mLastProcessedState==Tracking::NOT_INITIALIZED)
{
mvIniKeys=pTracker->mInitialFrame.mvKeys;
mvIniMatches=pTracker->mvIniMatches;
}
else if(pTracker->mLastProcessedState==Tracking::OK)
{
for(int i=0;i<N;i++)
{
MapPoint* pMP = pTracker->mCurrentFrame.mvpMapPoints[i];
if(pMP)
{
if(!pTracker->mCurrentFrame.mvbOutlier[i])
{
if(pMP->Observations()>0)
mvbMap[i]=true;
else
mvbVO[i]=true;
mmMatchedInImage[pMP->mnId] = mvCurrentKeys[i].pt;
}
else
{
mvpOutlierMPs.push_back(pMP);
mvOutlierKeys.push_back(mvCurrentKeys[i]);
}
}
}
}
mState=static_cast<int>(pTracker->mLastProcessedState);
}
} //namespace ORB_SLAM