forked from manhofer/Line3Dpp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
view.h
234 lines (189 loc) · 7.41 KB
/
view.h
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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
#ifndef I3D_LINE3D_PP_VIEW_H_
#define I3D_LINE3D_PP_VIEW_H_
/*
Line3D++ - Line-based Multi View Stereo
Copyright (C) 2015 Manuel Hofer
This program 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.
This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
*/
// check libs
#include "configLIBS.h"
// std
#include <map>
// external
#include "eigen3/Eigen/Eigen"
#include "boost/thread/mutex.hpp"
// opencv
#ifndef L3DPP_OPENCV3
#include "opencv/cv.h"
#else
#include "opencv2/imgproc.hpp"
#include "opencv2/core.hpp"
#endif //L3DPP_OPENCV3
// internal
#include "commons.h"
#include "dataArray.h"
#include "segment3D.h"
#include "cudawrapper.h"
/**
* Line3D++ - View Class
* ====================
* Holds all relevant data for one
* specific image.
* ====================
* Author: M.Hofer, 2016
*/
namespace L3DPP
{
class View
{
public:
View(const unsigned int id, L3DPP::DataArray<float4>* lines,
const Eigen::Matrix3d& K, const Eigen::Matrix3d& R,
const Eigen::Vector3d& t,
const unsigned int width, const unsigned int height,
const float median_depth,
L3DPP::DataArray<float>* superpixels=NULL);
~View();
// compute spatial regularizer (from pixel value)
void computeSpatialRegularizer(const float r);
float getSpecificSpatialReg(const float r);
// find collinear segments
void findCollinearSegments(const float dist_t, bool useGPU);
// draws lines into image
void drawLineImage(cv::Mat& img);
void drawSingleLine(const unsigned int id, cv::Mat& img,
const cv::Scalar& color);
void drawEpipolarLine(const Eigen::Vector3d& epi, cv::Mat& img);
// access to collinear segments
std::list<unsigned int> collinearSegments(const unsigned int segID);
// get ray from 2D point (normalized)
Eigen::Vector3d getNormalizedRay(const Eigen::Vector3d& p);
Eigen::Vector3d getNormalizedRay(const Eigen::Vector2d& p);
Eigen::Vector3d getNormalizedLinePointRay(const unsigned int lID,
const bool pt1);
// unproject 2D segment to 3D
L3DPP::Segment3D unprojectSegment(const unsigned int segID, const float depth1,
const float depth2);
// projects a 3D point into image
Eigen::Vector2d project(const Eigen::Vector3d& P);
Eigen::Vector3d projectWithCheck(const Eigen::Vector3d& P);
// get optical axis
Eigen::Vector3d getOpticalAxis();
// angle between views or view and segment (in rad)
double opticalAxesAngle(L3DPP::View* v);
double segmentQualityAngle(const L3DPP::Segment3D& seg3D,
const unsigned int segID);
// computes a projective visual neighbor score (to ensure bigger baselines)
float distanceVisualNeighborScore(L3DPP::View* v);
// baseline between views
float baseLine(L3DPP::View* v);
// checks if a projected 3D segment is long enough
bool projectedLongEnough(const L3DPP::Segment3D& seg3D);
// set new regularization depth
void update_median_depth(const float d,
const float sigmaP,
const float med_scene_depth)
{
median_depth_ = d;
if(sigmaP > 0.0f)
{
// fixed sigma
k_ = sigmaP/med_scene_depth;
}
median_sigma_ = k_*median_depth_;
}
// compute k when fixed sigmaP is used
void update_k(const float sigmaP, const float med_scene_depth)
{
k_ = sigmaP/med_scene_depth;
}
// compute regularizer with respect to given 3D point
float regularizerFrom3Dpoint(const Eigen::Vector3d& P);
// get coordinates of a specific line segment
Eigen::Vector4f getLineSegment2D(const unsigned int id);
// translate view by a fixed vector
void translate(const Eigen::Vector3d& t);
// data access
unsigned int id() const {return id_;}
Eigen::Vector3d C() const {return C_;}
Eigen::Matrix3d K() const {return K_;}
Eigen::Matrix3d Kinv() const {return Kinv_;}
Eigen::Matrix3d R() const {return R_;}
Eigen::Matrix3d Rt() const {return Rt_;}
Eigen::Matrix3d RtKinv() const {return RtKinv_;}
Eigen::Vector3d t() const {return t_;}
Eigen::Vector3d pp() const {return pp_;}
unsigned int width() const {return width_;}
unsigned int height() const {return height_;}
float diagonal() const {return diagonal_;}
L3DPP::DataArray<float4>* lines(){return lines_;}
L3DPP::DataArray<float>* superpixels(){return superpixels_;}
size_t num_lines() const {return lines_->width();}
float k() const {return k_;}
float median_depth() const {return median_depth_;}
float median_sigma() const {return median_sigma_;}
// lock/unlock view specific mutex
void lock_mutex(){mutex_.lock();}
void unlock_mutex(){mutex_.unlock();}
#ifdef L3DPP_CUDA
// GPU data
L3DPP::DataArray<float>* RtKinvGPU(){return RtKinv_DA_;}
float3 C_GPU() const {return C_f3_;}
#endif //L3DPP_CUDA
private:
// find collinear segments
void findCollinGPU();
void findCollinCPU();
// checks if a point is on a segment (only approximately!)
// Note: use only cor collinearity estimation!
bool pointOnSegment(const Eigen::Vector3d& p1, const Eigen::Vector3d& p2,
const Eigen::Vector3d& x);
// collinearity helper function: point to line distance 2D
float distance_point2line_2D(const Eigen::Vector3d& line, const Eigen::Vector3d& p);
// smaller angle between two lines [0,pi/2]
float smallerAngle(const Eigen::Vector2d& v1, const Eigen::Vector2d& v2);
// lines
L3DPP::DataArray<float4>* lines_;
// superpixels (Plane3D)
L3DPP::DataArray<float>* superpixels_;
// camera
unsigned int id_;
Eigen::Matrix3d K_;
Eigen::Matrix3d Kinv_;
Eigen::Matrix3d R_;
Eigen::Matrix3d Rt_;
Eigen::Matrix3d RtKinv_;
Eigen::Vector3d t_;
Eigen::Vector3d C_;
Eigen::Vector3d pp_;
unsigned int width_;
unsigned int height_;
float diagonal_;
float min_line_length_;
// regularizer
float k_;
float initial_median_depth_;
float median_depth_;
float median_sigma_;
// collinearity
float collin_t_;
std::vector<std::list<unsigned int> > collin_;
// mutex
boost::mutex mutex_;
#ifdef L3DPP_CUDA
// camera data (GPU)
L3DPP::DataArray<float>* RtKinv_DA_;
float3 C_f3_;
#endif //L3DPP_CUDA
};
}
#endif //I3D_LINE3D_PP_VIEW_H_