-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhelperTrackLastKeyFrame.m
104 lines (85 loc) · 4.14 KB
/
helperTrackLastKeyFrame.m
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
%helperTrackLastKeyFrame Estimate the camera pose by tracking the last key frame
% [currPose, mapPointIdx, featureIdx] = helperTrackLastKeyFrameStereo(mapPoints,
% views, currFeatures, currPoints, lastKeyFrameId, intrinsics) estimates
% the camera pose of the current frame by matching features with the
% previous key frame.
%
% This is an example helper function that is subject to change or removal
% in future releases.
%
% Inputs
% ------
% mapPoints - A helperMapPoints objects storing map points
% views - View attributes of key frames
% currFeatures - Features in the current frame
% currPoints - Feature points in the current frame
% lastKeyFrameId - ViewId of the last key frame
% intrinsics - Camera intrinsics
% scaleFactor - scale factor of features
%
% Outputs
% -------
% currPose - Estimated camera pose of the current frame
% mapPointIdx - Indices of map points observed in the current frame
% featureIdx - Indices of features corresponding to mapPointIdx
% Copyright 2019-2020 The MathWorks, Inc.
function [currPose, mapPointIdx, featureIdx] = helperTrackLastKeyFrame(...
mapPoints, views, currFeatures, currPoints, lastKeyFrameId, intrinsics, scaleFactor)
% Match features from the previous key frame with known world locations
[index3d, index2d] = findWorldPointsInView(mapPoints, lastKeyFrameId);
lastKeyFrameFeatures = views.Features{lastKeyFrameId}(index2d,:);
lastKeyFramePoints = views.Points{lastKeyFrameId}(index2d);
indexPairs = matchFeatures(currFeatures, binaryFeatures(lastKeyFrameFeatures),...
'Unique', true, 'MaxRatio', 0.9, 'MatchThreshold', 40);
% Estimate the camera pose
matchedImagePoints = currPoints.Location(indexPairs(:,1),:);
matchedWorldPoints = mapPoints.WorldPoints(index3d(indexPairs(:,2)), :);
matchedImagePoints = cast(matchedImagePoints, 'like', matchedWorldPoints);
[worldOri, worldLoc, inlier, status] = estimateWorldCameraPose(...
matchedImagePoints, matchedWorldPoints, intrinsics, ...
'Confidence', 95, 'MaxReprojectionError', 3, 'MaxNumTrials', 1e4);
if status
currPose=[];
mapPointIdx=[];
featureIdx=[];
return
end
currPose = rigid3d(worldOri, worldLoc);
% Refine camera pose only
currPose = bundleAdjustmentMotion(matchedWorldPoints(inlier,:), ...
matchedImagePoints(inlier,:), currPose, intrinsics, ...
'PointsUndistorted', true, 'AbsoluteTolerance', 1e-7,...
'RelativeTolerance', 1e-15, 'MaxIteration', 20);
% Search for more matches with the map points in the previous key frame
xyzPoints = mapPoints.WorldPoints(index3d,:);
[R, t] = cameraPoseToExtrinsics(currPose.Rotation, currPose.Translation);
[projectedPoints, isInImage] = worldToImage(intrinsics, R, t, xyzPoints);
projectedPoints = projectedPoints(isInImage, :);
minScales = max(1, lastKeyFramePoints.Scale(isInImage)/scaleFactor);
maxScales = lastKeyFramePoints.Scale(isInImage)*scaleFactor;
r = 4;
searchRadius = r*lastKeyFramePoints.Scale(isInImage);
indexPairs = matchFeaturesInRadius(binaryFeatures(lastKeyFrameFeatures(isInImage,:)), ...
binaryFeatures(currFeatures.Features), currPoints, projectedPoints, searchRadius, ...
'MatchThreshold', 40, 'MaxRatio', 0.8, 'Unique', true);
if size(indexPairs, 1) < 20
currPose=[];
mapPointIdx=[];
featureIdx=[];
return
end
% Filter by scales
isGoodScale = currPoints.Scale(indexPairs(:, 2)) >= minScales(indexPairs(:, 1)) & ...
currPoints.Scale(indexPairs(:, 2)) <= maxScales(indexPairs(:, 1));
indexPairs = indexPairs(isGoodScale, :);
% Obtain the index of matched map points and features
tempIdx = find(isInImage); % Convert to linear index
mapPointIdx = index3d(tempIdx(indexPairs(:,1)));
featureIdx = indexPairs(:,2);
% Refine the camera pose again
matchedWorldPoints = mapPoints.WorldPoints(mapPointIdx, :);
matchedImagePoints = currPoints.Location(featureIdx, :);
currPose = bundleAdjustmentMotion(matchedWorldPoints, matchedImagePoints, ...
currPose, intrinsics, 'PointsUndistorted', true, 'AbsoluteTolerance', 1e-7,...
'RelativeTolerance', 1e-15, 'MaxIteration', 20);
end