-
Notifications
You must be signed in to change notification settings - Fork 904
/
Copy pathexample_19-04.cpp
74 lines (66 loc) · 2.26 KB
/
example_19-04.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
// Example 19-4. Two-dimensional line fitting
#include <opencv2/opencv.hpp>
#include <iostream>
#include <math.h>
using namespace std;
void help(char **argv) {
cout << "\nExample 19-04, two dimensional line fitting"
<< "\nCall"
<< "\n" << argv[0] << "\n"
<< "\n 'q', 'Q' or ESC to quit"
<< "\n" << endl;
}
int main(int argc, char **argv) {
cv::Mat img(500, 500, CV_8UC3);
cv::RNG rng(-1);
help(argv);
for (;;) {
char key;
int i, count = rng.uniform(0, 100) + 3, outliers = count / 5;
float a = (float)rng.uniform(0., 200.);
float b = (float)rng.uniform(0., 40.);
float angle = (float)rng.uniform(0., CV_PI);
float cos_a = cos(angle), sin_a = sin(angle);
cv::Point pt1, pt2;
vector<cv::Point> points(count);
cv::Vec4f line;
float d, t;
b = MIN(a * 0.3f, b);
// generate some points that are close to the line
for (i = 0; i < count - outliers; i++) {
float x = (float)rng.uniform(-1., 1.) * a;
float y = (float)rng.uniform(-1., 1.) * b;
points[i].x = cvRound(x * cos_a - y * sin_a + img.cols / 2);
points[i].y = cvRound(x * sin_a + y * cos_a + img.rows / 2);
}
// generate outlier points
for (; i < count; i++) {
points[i].x = rng.uniform(0, img.cols);
points[i].y = rng.uniform(0, img.rows);
}
// find the optimal line
cv::fitLine(points, line, cv::DIST_L1, 1, 0.001, 0.001);
// draw the points
img = cv::Scalar::all(0);
for (i = 0; i < count; i++)
cv::circle(img, points[i], 2,
i < count - outliers ? cv::Scalar(0, 0, 255)
: cv::Scalar(0, 255, 255),
cv::FILLED, CV_AA, 0);
// ... and the long enough line to cross the whole image
d = sqrt((double)line[0] * line[0] + (double)line[1] * line[1]);
line[0] /= d;
line[1] /= d;
t = (float)(img.cols + img.rows);
pt1.x = cvRound(line[2] - line[0] * t);
pt1.y = cvRound(line[3] - line[1] * t);
pt2.x = cvRound(line[2] + line[0] * t);
pt2.y = cvRound(line[3] + line[1] * t);
cv::line(img, pt1, pt2, cv::Scalar(0, 255, 0), 3, CV_AA, 0);
cv::imshow("Fit Line", img);
key = (char)cv::waitKey(0);
if (key == 27 || key == 'q' || key == 'Q') // 'ESC'
break;
}
return 0;
}