opencv2 使用鼠标绘制矩形并截取和保存矩形区域图像

  前言  

     好长时间没写博文了,今天偷偷懒写篇关于opencv2中鼠标响应操作的文章。

      鼠标操作属于用户接口设计,以前一直使用Qt来做,但是如果只需要简单的鼠标,键盘操作,直接调用opencv库的函数也未尝不可,鼠标操作之前已经接触很多了,在MFC,QT,OpenGL,等等中,理论主要就是两点,一是监控鼠标操作,鼠标点击,移动,松开,然后通过mouse_event识别判断出那一种鼠标的操作,根据不同的操作然后进行处理,二是在主函数中加入鼠标的回调函数,将鼠标操作与程序的窗口绑定。

                                              第一节 函数介绍

       暂时只接触了两个关于opencv2鼠标响应操作的函数,下面分别介绍一下:

    1.1 回调函数

      opencv2.4.5中,提供的鼠标回调函数是 setMouseCallback,函数声明如下:

[cpp] view plaincopy

  1. CV_EXPORTS void setMouseCallback(const string& winname, MouseCallback onMouse, void* userdata = 0);  

     函数参数介绍

            const string& winname,windows视窗名称,对名为winname的视窗进行鼠标监控。

            MouseCallback onMouse,鼠标响应处理函数,监听鼠标的点击,移动,松开,判断鼠标的操作类型,并进行响应的函数处理。

            void* userdata = 0 鼠标响应处理函数的ID,与鼠标相应处理函数相匹配就行,暂时只用到默认为0的情况。

     函数使用实例:

[cpp] view plaincopy

  1. namedWindow("img");  
  2. setMouseCallback("img",on_mouse,0);  

    1.2 鼠标响应处理函数

       opencv2.4.5中,鼠标相应处理函数一般默认形参和返回参数,函数形式如下:

[cpp] view plaincopy

  1. void on_mouse(int event,int x,int y,int flags,void *ustc)  

      函数参数介绍:

      int event,鼠标操作时间的整数代号,在opencv2.4.5中,event鼠标事件总共有10中,从0-9依次代表如下:

Event:

[cpp] view plaincopy

  1. #define CV_EVENT_MOUSEMOVE 0             滑动  
  2. #define CV_EVENT_LBUTTONDOWN 1           左键点击  
  3. #define CV_EVENT_RBUTTONDOWN 2           右键点击  
  4. #define CV_EVENT_MBUTTONDOWN 3           中间点击  
  5. #define CV_EVENT_LBUTTONUP 4             左键释放  
  6. #define CV_EVENT_RBUTTONUP 5             右键释放  
  7. #define CV_EVENT_MBUTTONUP 6             中间释放  
  8. #define CV_EVENT_LBUTTONDBLCLK 7         左键双击  
  9. #define CV_EVENT_RBUTTONDBLCLK 8         右键双击  
  10. #define CV_EVENT_MBUTTONDBLCLK 9         中间释放  

     int x,int y,代表鼠标位于窗口的(x,y)坐标位置,窗口左上角默认为原点,向右为x轴,向下为y轴,

     int flags,代表鼠标的拖拽事件,以及键盘鼠标联合事件,总共有32种事件,依次如下:

flags:

[cpp] view plaincopy

  1. #define CV_EVENT_FLAG_LBUTTON 1           左键拖拽  
  2. #define CV_EVENT_FLAG_RBUTTON 2           右键拖拽  
  3. #define CV_EVENT_FLAG_MBUTTON 4           中间拖拽  
  4. #define CV_EVENT_FLAG_CTRLKEY 8     (8~15)按Ctrl不放事件  
  5. #define CV_EVENT_FLAG_SHIFTKEY 16   (16~31)按Shift不放事件  
  6. #define CV_EVENT_FLAG_ALTKEY 32       (32~39)按Alt不放事件(后面8-39还有待研究)  

    void *ustc,函数参数的编号(暂时用不到)

                                        第二节 鼠标操作实例

        2.1 示例程序代码

        程序如下,已经附上说明:

[cpp] view plaincopy

  1. #include <opencv2/core/core.hpp>  
  2. #include <opencv2/highgui/highgui.hpp>  
  3. #include <stdio.h>  
  4.   
  5. using namespace cv;  
  6.   
  7. cv::Mat org,dst,img,tmp;  
  8. void on_mouse(int event,int x,int y,int flags,void *ustc)//event鼠标事件代号,x,y鼠标坐标,flags拖拽和键盘操作的代号  
  9. {  
  10.     static Point pre_pt = (-1,-1);//初始坐标  
  11.     static Point cur_pt = (-1,-1);//实时坐标  
  12.     char temp[16];  
  13.     if (event == CV_EVENT_LBUTTONDOWN)//左键按下,读取初始坐标,并在图像上该点处划圆  
  14.     {  
  15.         org.copyTo(img);//将原始图片复制到img中  
  16.         sprintf(temp,"(%d,%d)",x,y);  
  17.         pre_pt = Point(x,y);  
  18.         putText(img,temp,pre_pt,FONT_HERSHEY_SIMPLEX,0.5,Scalar(0,0,0,255),1,8);//在窗口上显示坐标  
  19.         circle(img,pre_pt,2,Scalar(255,0,0,0),CV_FILLED,CV_AA,0);//划圆  
  20.         imshow("img",img);  
  21.     }  
  22.     else if (event == CV_EVENT_MOUSEMOVE && !(flags & CV_EVENT_FLAG_LBUTTON))//左键没有按下的情况下鼠标移动的处理函数  
  23.     {  
  24.         img.copyTo(tmp);//将img复制到临时图像tmp上,用于显示实时坐标  
  25.         sprintf(temp,"(%d,%d)",x,y);  
  26.         cur_pt = Point(x,y);  
  27.         putText(tmp,temp,cur_pt,FONT_HERSHEY_SIMPLEX,0.5,Scalar(0,0,0,255));//只是实时显示鼠标移动的坐标  
  28.         imshow("img",tmp);  
  29.     }  
  30.     else if (event == CV_EVENT_MOUSEMOVE && (flags & CV_EVENT_FLAG_LBUTTON))//左键按下时,鼠标移动,则在图像上划矩形  
  31.     {  
  32.         img.copyTo(tmp);  
  33.         sprintf(temp,"(%d,%d)",x,y);  
  34.         cur_pt = Point(x,y);  
  35.         putText(tmp,temp,cur_pt,FONT_HERSHEY_SIMPLEX,0.5,Scalar(0,0,0,255));  
  36.         rectangle(tmp,pre_pt,cur_pt,Scalar(0,255,0,0),1,8,0);//在临时图像上实时显示鼠标拖动时形成的矩形  
  37.         imshow("img",tmp);  
  38.     }  
  39.     else if (event == CV_EVENT_LBUTTONUP)//左键松开,将在图像上划矩形  
  40.     {  
  41.         org.copyTo(img);  
  42.         sprintf(temp,"(%d,%d)",x,y);  
  43.         cur_pt = Point(x,y);  
  44.         putText(img,temp,cur_pt,FONT_HERSHEY_SIMPLEX,0.5,Scalar(0,0,0,255));  
  45.         circle(img,pre_pt,2,Scalar(255,0,0,0),CV_FILLED,CV_AA,0);  
  46.         rectangle(img,pre_pt,cur_pt,Scalar(0,255,0,0),1,8,0);//根据初始点和结束点,将矩形画到img上  
  47.         imshow("img",img);  
  48.         img.copyTo(tmp);  
  49.         //截取矩形包围的图像,并保存到dst中  
  50.         int width = abs(pre_pt.x - cur_pt.x);  
  51.         int height = abs(pre_pt.y - cur_pt.y);  
  52.         if (width == 0 || height == 0)  
  53.         {  
  54.             printf("width == 0 || height == 0");  
  55.             return;  
  56.         }  
  57.         dst = org(Rect(min(cur_pt.x,pre_pt.x),min(cur_pt.y,pre_pt.y),width,height));  
  58.         namedWindow("dst");  
  59.         imshow("dst",dst);  
  60.         waitKey(0);  
  61.     }  
  62. }  
  63. void main()  
  64. {  
  65.     org = imread("1.jpg");  
  66.     org.copyTo(img);  
  67.     org.copyTo(tmp);  
  68.     namedWindow("img");//定义一个img窗口  
  69.     setMouseCallback("img",on_mouse,0);//调用回调函数  
  70.     imshow("img",img);  
  71.     cv::waitKey(0);  
  72. }  

     2.2 程序运行结果分析

     程序源图形:

     

   程序运行结果:

    

                                            第三节,参考资料

         既然参考别人的博文,当然要把源资料介绍给大家,让大家参考一下:

       1. 在OpenCV中利用鼠标绘制矩形和截取图像的矩形区域 

                                        http://blog.csdn.net/quarryman/article/details/6435527  

      2.OpenCV响应鼠标函数cvSetMouseCallback()和其副程式onMouse()的使用(OpenCV2.4.5)

                                      http://blog.csdn.net/glb562000520/article/details/8938582

时间: 2024-12-04 13:35:05

opencv2 使用鼠标绘制矩形并截取和保存矩形区域图像的相关文章

在OpenCV中利用鼠标绘制矩形和截取图像的矩形区域

这是两个相关的程序,前者是后者的基础.实际上前一个程序也是在前面博文的基础上做的修改,请参考<在OpenCV中利用鼠标绘制直线> .下面贴出代码. 程序之一,在OpenCV中利用鼠标绘制矩形 [c-sharp] view plaincopy #include <cv.h>   #include <highgui.h>   #include <stdio.h>   #pragma comment( lib, "cv.lib" )   #pr

Illustrator鼠标绘制一个精美的药箱图标教程

给各位Illustrator软件的使用者们来详细的解析分享一下鼠标绘制一个精美的药箱图标的教程. 教程分享: 1.新建文件,画个框(矩形工具快捷键M),填红色.   2.对刚才的矩形用"效果-风格化-圆角",然后"对象-展开外观",这样就得到一个圆角矩形,复制一份,以备不时之需.(我就纳闷了,这外国人咋就不直接用圆角矩形来画这个呢?)   3.选中刚才得到的圆角矩形,用"效果-三维-突出和斜角",调整下角度和透视的大小以及灯光方向和强度等相关选项

photoshop鼠标绘制蜡笔风格唇膏效果教程

给各位photoshop软件的使用者们来详细的解析分享一下鼠标绘制蜡笔风格唇膏效果的教程. 教程分享: 最终效果     1.对要绘画的对象大概的分析,对基本型.型的组合.颜色的搭配有个概念.   2.选用"干画笔尖浅描"笔刷.   3.设置画笔透明度.   4.新建一个图层,先用圆角矩形工具,画第一个型,然后,再建一个新图层.按Alt + 鼠标左键,点击图层1,建立一个跟圆角矩形一样的选区,再点击图层2.             5.用画笔涂抹出基本型.     6.增加立体感,涂抹

c++-鼠标绘制曲线,解码恢复,最好用VSC#

问题描述 鼠标绘制曲线,解码恢复,最好用VSC# 手绘单条曲线(横坐标为时间轴),读取鼠标事件获得坐标值,设置采样间隔,将曲线 进行采样.量化.编码并存储.可读入已有编码数据并描点连线恢复原曲线. 解决方案 使用chart控件 http://blog.sina.com.cn/s/blog_621e24e20101cp64.html

java-JAVA百度地图鼠标绘制工具获取坐标保存数据库

问题描述 JAVA百度地图鼠标绘制工具获取坐标保存数据库 百度地图绘制工具里面绘制的图案或者覆盖物,怎么获取到坐标保存到数据库下次加载就数据,页面直接显示. 解决方案 http://blog.csdn.net/dyllove98/article/details/9069889 解决方案二: 首先根据百度地图API调用百度地图获取坐标信息,其次就是用java进行数据存储了. 参考:http://blog.csdn.net/bailin0007/article/details/11973401

photoshop鼠标绘制一只卡通熊猫制作教程

给各位photoshop软件的使用者们来详细的解析分享一下鼠标绘制一只卡通熊猫的制作教程. 教程分享: 1.没错,直接椭圆工具画个圆形.     2.调整一下图形,至少看起来像个脸(虽然更像包子).     3.耳朵同样是使用椭圆工具.     4.把耳朵放在合适的位置,貌似熊猫已经画完了.     5.熬夜的黑眼圈是熊猫最大的特征吧.     6.眼睛+鼻子,感觉眼睛被掏空-至此为止都是椭圆+椭圆+椭圆.     7.这一步是眉毛+嘴巴,主要也是为了说明我还是会用钢笔.     8.眼睛是最重

photoshop鼠标绘制抽象光束效果教程分享

给各位photoshop软件的使用者们来详细的解析分享一下鼠标绘制抽象光束效果的教程. 教程分享: 1.我们先来制作花纹的最基本的构成部分,如下图.   <图1> 2.新建一个1024* 768像素的文档,背景填充黑色.新建一个组,用钢笔勾出下图所示的选区,给组添加蒙版.   <图2> 3.在组里新建一个图层,选择渐变工具,颜色设置如图3,由右上角向左下角拉出线性渐变,再把图层不透明度改为:20%,效果如图4.   <图3>   <图4> 4.用钢笔勾出底部

python opencv 双击鼠标绘制圆

10-python opencv 双击鼠标绘制圆 10-python opencv 双击鼠标绘制圆 概述 实现过程 引用与创建空图 设置回调函数 回调上述函数 显示图像 源代码 运行结果 参考 概述 本节实现的是使用OpenCV里自带的函数,在双击图片时,以其为圆心绘制圆. 回调函数 捕捉鼠标事件 实现过程 引用与创建空图 不再赘述,代码如下. import cv2 import numpy # empty image img = np.zeros((512, 512, 3), np.uint8

Illustrator鼠标绘制3D渲染汽车教程

给各位Illustrator软件的使用者们来详细的解析分享一下鼠标绘制3D渲染汽车的教程. 教程分享: 1.首先把整个车子大形全部用钢笔工具勾勒出来.   图02 2.整个车子大形全部用形状色块区分出来.   图03 3.最重要的部分来了!下面就是细化结构.主要运用了网格工具!   图04   图05 网格工具,刚开始用的时候比较生硬难以把控,只要多用用多练练就熟能生巧了,做出的效果不比ps渲染的差. 运用网格工具的小技巧:多余的网格可以按住"alt"键删掉,要想颜色晕染的真实,网格不