-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathTarget.cpp
164 lines (146 loc) · 5.28 KB
/
Target.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
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
#include "niVision.h"
#include "Vision/ColorImage.h"
#include "Vision/MonoImage.h"
#include "Target.h"
#include <algorithm>
#include <math.h>
// These parameters set ellipse finding in the NI imaq (Image Aquisition) library.
// Refer to the CVI Function Reference PDF document installed with LabVIEW for
// additional information.
static EllipseDescriptor ellipseDescriptor = {
3, // minMajorRadius
200, // maxMajorRadius
3, // minMinorRadius
100 // maxMinorRadius
};
static CurveOptions curveOptions = { IMAQ_NORMAL_IMAGE, // extractionMode
40, // threshold
IMAQ_NORMAL, // filterSize
25, // minLength
15, // rowStepSize
15, // columnStepSize
10, // maxEndPointGap
1, // onlyClosed
0 // subpixelAccuracy
};
static ShapeDetectionOptions shapeOptions = {
IMAQ_GEOMETRIC_MATCH_SHIFT_INVARIANT, // mode
NULL, // angle ranges
0, // num angle ranges
0, // scale range
500 // minMatchScore
};
double Target::GetHorizontalAngle()
{
double x = m_xPos;
x = x * 9.0 / m_xMax;
x = atan2(x, 20.0);
x = x * 180.0 / 3.14159;
return x;
}
/**
* Compare two targets.
* Compare the score of two targets for the sort function in C++.
* @param t1 the first Target
* @param t2 the second Target
* @returns (1, 0, or -1) for the scores of t1 > t2, t1 == t2, and t1 < t2
**/
int compareTargets(Target t1, Target t2)
{
if (t1.m_score > t2.m_score) return 1;
if (t1.m_score < t2.m_score) return -1;
return 0;
}
/**
* Find the best circular target in the image.
* The best target is the one with two concentric circles, largest size, and best score
* @param image The image to examine.
* @returns A target object with the parameters filled in.
*/
vector<Target> Target::FindCircularTargets(ColorImage *image)
{
wpi_assert(image != NULL);
int width = image->GetWidth();
int height = image->GetHeight();
vector<Target> sortedTargets;
// get the luminance plane only for the image to make the code
// insensitive to lighting conditions.
MonoImage *luminancePlane = image->GetLuminancePlane();
vector<EllipseMatch> *results = luminancePlane->DetectEllipses(&ellipseDescriptor,
&curveOptions,
&shapeOptions,
NULL);
delete luminancePlane;
if (results->size() == 0)
{
return sortedTargets;
}
// create a list of targets corresponding to each ellipse found
// in the image.
for (unsigned i = 0; i < results->size(); i++)
{
Target target;
EllipseMatch e = results->at(i);
target.m_rawScore = e.score;
target.m_score = (e.majorRadius * e.minorRadius)
/ (1001 - e.score)
/ (height * width) * 100;
target.m_majorRadius = e.majorRadius / height;
target.m_minorRadius = e.minorRadius / height;
//always divide by height so that x and y are same units
target.m_xPos = (2.0 * e.position.x - width) / height;
target.m_yPos = (2.0 * e.position.y - height) / height;
target.m_rotation = e.rotation;
target.m_xMax = (double)width / height;
target.m_bothFound = false;
sortedTargets.push_back(target);
}
delete results;
// sort the list of targets by score
sort(sortedTargets.begin(), sortedTargets.end(), compareTargets);
// go through each target found in descending score order and look
// for another target whose center is contained inside of this target
// Those concentric targets get a score which is the sum of both targets
vector<Target> combinedTargets;
while (sortedTargets.size() > 0)
{
vector<Target>::iterator iter = sortedTargets.begin();
Target t1 = *iter++;
for (; iter < sortedTargets.end(); iter++)
{
Target t2 = *iter;
// check if the two are concentric
if ((fabs(t1.m_xPos - t2.m_xPos) < min(t1.m_minorRadius, t2.m_minorRadius)) &&
(fabs(t1.m_yPos - t2.m_yPos) < min(t1.m_majorRadius, t2.m_majorRadius)))
{
// create the information for the combined target
// (the 2 concentric ellipses)
t1.m_xPos = (t1.m_xPos + t2.m_xPos) / 2;
t1.m_yPos = (t1.m_yPos + t2.m_yPos) / 2;
t1.m_rawScore += t2.m_rawScore;
t1.m_score += t2.m_score;
t1.m_majorRadius = max(t1.m_majorRadius, t2.m_majorRadius);
t1.m_minorRadius = max(t1.m_minorRadius, t2.m_minorRadius);
t1.m_bothFound = true;
sortedTargets.erase(iter); // loop needs to be an interator
break;
}
}
sortedTargets.erase(sortedTargets.begin());
combinedTargets.push_back(t1);
}
// sort the combined targets so the highest scoring one is first
sort(combinedTargets.begin(), combinedTargets.end(), compareTargets);
return combinedTargets;
}
/**
* Print the target.
* Print information about this target object.
*/
void Target::Print()
{
printf("Target found: (%f, %f), major: %f, minor: %f\nrotation: %f, score: %f, both: %d\n\n",
m_xPos, m_yPos,
m_majorRadius, m_minorRadius,
m_rotation, m_score, m_bothFound);
}