#include <iostream>
class canvas{
public:
bool setupQ;
int minDims,maxDims;
double scale;
int rows, cols;
void init(int minD, int maxD){
minDims = minD; maxDims = maxD;
scale = 1.0;
rows = 0;
cols = 0;
setupQ = false;
}
if(setupQ){
if(corner.
x < max.x){corner.
x = (int)(max.x + 1.0);};
if(corner.
y < max.y){corner.
y = (int)(max.y + 1.0);};
if(origin.
x > min.x){origin.
x = (int) min.x;};
if(origin.
y > min.y){origin.
y = (int) min.y;};
} else {
corner =
cv::Point((
int)(max.x + 1.0), (
int)(max.y + 1.0));
}
int c = (int)(scale*((corner.
x + 1.0) - origin.
x));
if(c<minDims){
scale = scale * (double)minDims/(double)c;
} else {
if(c>maxDims){
scale = scale * (double)maxDims/(double)c;
}
}
int r = (int)(scale*((corner.
y + 1.0) - origin.
y));
if(r<minDims){
scale = scale * (double)minDims/(double)r;
} else {
if(r>maxDims){
scale = scale * (double)maxDims/(double)r;
}
}
cols = (int)(scale*((corner.
x + 1.0) - origin.
x));
rows = (int)(scale*((corner.
y + 1.0) - origin.
y));
setupQ = true;
}
void stretch(vector<Point2f> pts)
{
for(size_t i=1; i < pts.size(); i++){
if(max.x < pnt.
x){max.x = pnt.
x;};
if(max.y < pnt.
y){max.y = pnt.
y;};
if(min.x > pnt.
x){min.x = pnt.
x;};
if(min.y > pnt.
y){min.y = pnt.
y;};
};
stretch(min, max);
}
{
for( int i = 0; i < 4; i++ ){
if(max.x < pnt.
x){max.x = pnt.
x;};
if(max.y < pnt.
y){max.y = pnt.
y;};
if(min.x > pnt.
x){min.x = pnt.
x;};
if(min.y > pnt.
y){min.y = pnt.
y;};
}
stretch(min, max);
}
{
stretch(box);
}
ellipse(img, box, color, lineThickness, LINE_AA);
for( int j = 0; j < 4; j++ ){
line(img, vtx[j], vtx[(j+1)%4], color, lineThickness, LINE_AA);
}
}
void drawPoints(vector<Point2f> pts,
cv::Scalar color)
{
stretch(pts);
}
for(size_t i=0; i < pts.size(); i++){
};
}
void drawLabels( std::vector<std::string> text, std::vector<cv::Scalar> colors)
{
}
int vPos = 0;
for (size_t i=0; i < text.size(); i++) {
std::string txt = text[i];
Size textsize = getTextSize(txt, FONT_HERSHEY_COMPLEX, 1, 1, 0);
vPos += (int)(1.3 * textsize.
height);
cv::putText(img, txt, org, FONT_HERSHEY_COMPLEX, 1, color, 1, LINE_8);
}
}
};
static void help(char** argv)
{
cout << "\n该程序演示了椭圆拟合。程序查找\n"
轮廓并用椭圆对其进行近似。使用三种方法查找\n
椭圆拟合:fitEllipse、fitEllipseAMS 和 fitEllipseDirect。\n
调用:\n
<< argv[0] << " [图像名称 -- 默认 ellipses.jpg]\n" << endl;
}
int sliderPos = 70;
bool fitEllipseQ, fitEllipseAMSQ, fitEllipseDirectQ;
void processImage(int, void*);
int main(
int argc,
char** argv )
{
fitEllipseQ = true;
fitEllipseAMSQ = true;
fitEllipseDirectQ = true;
if (parser.has("help"))
{
help(argv);
return 0;
}
string filename = parser.get<string>("@image");
image =
imread(samples::findFile(filename), 0);
if( image.empty() )
{
cout << "无法打开图像 " << filename << "\n";
return 0;
}
createTrackbar(
"threshold",
"result", &sliderPos, 255, processImage );
processImage(0, 0);
return 0;
}
}
void processImage(int , void*)
{
vector<vector<Point> > contours;
Mat bimage = image >= sliderPos;
findContours(bimage, contours, RETR_LIST, CHAIN_APPROX_NONE);
canvas paper;
std::vector<std::string> text;
std::vector<cv::Scalar> color;
if (fitEllipseQ) {
text.push_back("OpenCV");
color.push_back(fitEllipseColor);
}
if (fitEllipseAMSQ) {
text.push_back("AMS");
color.push_back(fitEllipseAMSColor);
}
if (fitEllipseDirectQ) {
text.push_back("Direct");
color.push_back(fitEllipseDirectColor);
}
paper.drawLabels(text, color);
int margin = 2;
vector< vector<Point2f> > points;
for(size_t i = 0; i < contours.size(); i++)
{
size_t count = contours[i].size();
if( count < 6 )
continue;
vector<Point2f>pts;
for (
int j = 0; j < pointsf.
rows; j++) {
if ((pnt.
x > margin && pnt.
y > margin && pnt.
x < bimage.
cols-margin && pnt.
y < bimage.
rows-margin)) {
if(j%20==0){
pts.push_back(pnt);
}
}
}
points.push_back(pts);
}
for(size_t i = 0; i < points.size(); i++)
{
vector<Point2f> pts = points[i];
if (pts.size()<5) {
continue;
}
if (fitEllipseQ) {
if (isGoodBox(box)) {
paper.drawEllipseWithBox(box, fitEllipseColor, 3);
}
}
if (fitEllipseAMSQ) {
if (isGoodBox(boxAMS)) {
paper.drawEllipseWithBox(boxAMS, fitEllipseAMSColor, 2);
}
}
if (fitEllipseDirectQ) {
if (isGoodBox(boxDirect)){
paper.drawEllipseWithBox(boxDirect, fitEllipseDirectColor, 1);
}
}
paper.drawPoints(pts, fitEllipseTrueColor);
}
}
用于命令行解析。
定义 utility.hpp:890
static CV_NODISCARD_STD MatExpr zeros(int rows, int cols, int type)
返回指定大小和类型的零数组。
_Tp & at(int i0=0)
返回对指定数组元素的引用。
bool empty() const
如果数组没有元素,则返回 true。
int rows
行和列的数量,当矩阵的维度大于 2 时为 (-1, -1)
定义 mat.hpp:2155
void convertTo(OutputArray m, int rtype, double alpha=1, double beta=0) const
将数组转换为另一种数据类型,并可选缩放。
_Tp y
点的 y 坐标
定义 types.hpp:202
_Tp x
点的x坐标
定义 types.hpp:201
该类表示平面上的旋转矩形(即非正矩形)。
定义 types.hpp:538
Point2f center
返回矩形的质心
定义 types.hpp:570
Size2f size
返回矩形的宽度和高度
定义 types.hpp:572
void points(Point2f pts[]) const
用于指定图像或矩形大小的模板类。
定义 types.hpp:335
_Tp height
高度
定义 types.hpp:363
_Tp width
宽度
定义 types.hpp:362
短数值向量的模板类,Matx 的一种特例。
定义 matx.hpp:369
Point2i Point
定义 types.hpp:209
Point_< float > Point2f
定义 types.hpp:207
#define CV_32F
定义 interface.h:78
unsigned char uchar
定义 interface.h:51
#define CV_8UC3
定义 interface.h:90
#define MIN(a, b)
定义 cvdef.h:514
#define MAX(a, b)
定义 cvdef.h:518
void imshow(const String &winname, InputArray mat)
在指定的窗口中显示图像。
int waitKey(int delay=0)
等待按键按下。
void namedWindow(const String &winname, int flags=WINDOW_AUTOSIZE)
创建一个窗口。
int createTrackbar(const String &trackbarname, const String &winname, int *value, int count, TrackbarCallback onChange=0, void *userdata=0)
创建一个轨迹条并将其附加到指定的窗口。
CV_EXPORTS_W Mat imread(const String &filename, int flags=IMREAD_COLOR_BGR)
从文件中加载图像。
void putText(InputOutputArray img, const String &text, Point org, int fontFace, double fontScale, Scalar color, int thickness=1, int lineType=LINE_8, bool bottomLeftOrigin=false)
绘制文本字符串。
RotatedRect fitEllipseDirect(InputArray points)
围绕一组二维点拟合椭圆。
RotatedRect fitEllipseAMS(InputArray points)
围绕一组二维点拟合椭圆。
void findContours(InputArray image, OutputArrayOfArrays contours, OutputArray hierarchy, int mode, int method, Point offset=Point())
在二值图像中查找轮廓。
RotatedRect fitEllipse(InputArray points)
围绕一组二维点拟合椭圆。
int main(int argc, char *argv[])
定义 highgui_qt.cpp:3