您的位置:首页 > 运维架构

opencv中实现鼠标选定矩形框

2013-05-29 09:14 597 查看

从网上找了一个opencv中实现鼠标选定视频矩形框,先转了以供后续研究

视频矩形框

#include "stdafx.h"
#include "opencv2/imgproc/imgproc.hpp"
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/core/core.hpp>
#include <stdio.h>
#include <iostream>

using namespace cv;
using namespace std;
Rect select;
bool select_flag=false;
Point origin;
Mat frame;

/************************************************************************************************************************/
/****                    如果采用这个onMouse()函数的话,则只能画出从左上到右下,或者从右下到左上的矩形框                    ****/
/************************************************************************************************************************/
//void onMouse(int event,int x,int y,int,void*)
//{
//    if(event==CV_EVENT_LBUTTONDOWN)
//    {
//        select.x=x;
//        select.y=y;
//        tracking=false;
//    }
//    else if(event==CV_EVENT_LBUTTONUP)
//    {
//        select.width=x-select.x;//以下2行计算出来的值要么都大于0,要么都小于0
//        select.height=y-select.y;
//        tracking=true;//左键完后,开始跟踪
//    }
//}

/************************************************************************************************************************/
/****                            如果采用这个onMouse()函数的话,则可以画出鼠标拖动矩形框的4种情形                        ****/
/************************************************************************************************************************/
void onMouse(int event,int x,int y,int,void*)
{
//Point origin;//不能在这个地方进行定义,因为这是基于消息响应的函数,执行完后origin就释放了,所以达不到效果。
if(select_flag)
{
select.x=MIN(origin.x,x);//不一定要等鼠标弹起才计算矩形框,而应该在鼠标按下开始到弹起这段时间实时计算所选矩形框
select.y=MIN(origin.y,y);
select.width=abs(x-origin.x);//算矩形宽度和高度
select.height=abs(y-origin.y);
select&=Rect(0,0,frame.cols,frame.rows);//保证所选矩形框在视频显示区域之内
}
if(event==CV_EVENT_LBUTTONDOWN)
{
select_flag=true;//鼠标按下的标志赋真值
origin=Point(x,y);//保存下来单击是捕捉到的点
select=Rect(x,y,0,0);//这里一定要初始化,宽和高为(0,0)是因为在opencv中Rect矩形框类内的点是包含左上角那个点的,但是不含右下角那个点
}
else if(event==CV_EVENT_LBUTTONUP)
{
select_flag=false;
}
}

int main(int argc, unsigned char* argv[])
{
char c;

//打开摄像头
VideoCapture cam(0);
if (!cam.isOpened())
return -1;

//建立窗口
namedWindow("camera",1);//显示视频原图像的窗口

//捕捉鼠标
setMouseCallback("camera",onMouse,0);

while(1)
{
//读取一帧图像
cam>>frame;
if(frame.empty())
return -1;

//画出矩形框
rectangle(frame,select,Scalar(255,0,0),3,8,0);//能够实时显示在画矩形窗口时的痕迹

//显示视频图片到窗口
imshow("camera",frame);

//    select.zeros();
//键盘响应
c=(char)waitKey(20);
if(27==c)//ESC键
return -1;
}

return 0;
}

图片矩形框

后期进行了整理,编写了图片上选定矩形框并显示新图片并保存,核心代码如下:

// Rect_mouse.cpp : 定义控制台应用程序的入口点。
//
#include "stdafx.h"
#include "opencv2/imgproc/imgproc.hpp"
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/core/core.hpp>
#include <stdio.h>
#include <iostream>
using namespace cv;

unsigned int count=1;
Rect select;
bool select_flag=false;
Point origin;
IplImage *src=NULL;

/************************************************************************************************************************/
/****                            如果采用这个onMouse()函数的话,则可以画出鼠标拖动矩形框的4种情形                        ****/
/************************************************************************************************************************/
void onMouse(int event,int x,int y,int,void*)
{
//Point origin;//不能在这个地方进行定义,因为这是基于消息响应的函数,执行完后origin就释放了,所以达不到效果。

if(event==CV_EVENT_LBUTTONDOWN)
{
select_flag=true;//鼠标按下的标志赋真值
origin=Point(x,y);//保存下来单击是捕捉到的点
select=Rect(x,y,0,0);//这里一定要初始化,宽和高为(0,0)是因为在opencv中Rect矩形框类内的点是包含左上角那个点的,但是不含右下角那个点
}
else if(event==CV_EVENT_LBUTTONUP)
{
select_flag=false;
}
if(select_flag)
{
select.x=MIN(origin.x,x);//不一定要等鼠标弹起才计算矩形框,而应该在鼠标按下开始到弹起这段时间实时计算所选矩形框
select.y=MIN(origin.y,y);
select.width=abs(x-origin.x);//算矩形宽度和高度
select.height=abs(y-origin.y);
select&=Rect(0,0,src->width,src->height);//保证所选矩形框在视频显示区域之内
}
}

IplImage* cvGetSubImage(IplImage *image, CvRect roi)
{
IplImage *result;
// 设置 ROI
cvSetImageROI(image,roi);
// 创建子图像
result = cvCreateImage( cvSize(roi.width, roi.height), image->depth, image->nChannels );
cvCopy(image,result);
cvResetImageROI(image);
return result;
}

int _tmain(int argc, _TCHAR* argv[])
{
char c;

src=cvLoadImage("Capture0.bmp");
if (!src)
{
return -1;
}
IplImage *dst=cvCloneImage(src);
while(1)
{
IplImage * temp= cvCreateImage(cvGetSize(src),src->depth,3);
cvZero(temp);
setMouseCallback("select roi",onMouse,0);

cvDrawRect(src,cvPoint(select.x,select.y),cvPoint(select.x+select.width,select.y+select.height),Scalar(255,255,255),1,8,0);//能够实时显示在画矩形窗口时的痕迹
cvShowImage("select roi",src);

c=(char)waitKey(20);
if(c==13)
{
cvDestroyWindow("roi");
cvNamedWindow("roi");
//一开始总选不了单独存图片因为roi是实时变化的  所以不行,现在改为回车即可
IplImage *roi=cvCreateImage(cvSize(select.width,select.height),8,3);
// 设置 ROI
cvZero(roi);
cvSetImageROI(src,cvRect(select.x,select.y,select.width,select.height));
// 创建子图像
roi = cvCreateImage( cvSize(select.width, select.height), src->depth, src->nChannels );
cvCopy(src,roi);
cvResetImageROI(src);
cvNamedWindow( "roi");
cvShowImage("roi",roi);
char *pic_name=new char[30];
memset(pic_name,NULL,sizeof(char)*30);
sprintf_s(pic_name,30,"%dth_",count);
strcat_s(pic_name,30,"roi");
strcat_s(pic_name,30,".bmp");
cvSaveImage(pic_name,roi);
cvReleaseImage(&roi);
count++;
}
if(c==27)//ESC键
{
break;
}
cvReleaseImage(&temp);
cvCopy(dst,src);
}
cvReleaseImage(&src);
cvReleaseImage(&dst);
cvDestroyWindow("camera");
cvDestroyWindow("roi");
cvDestroyWindow("temp");
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: