Opencv实现鼠标截图功能

 1.坐标系

首先我们先认识一下opencv的坐标系,我们的图片坐标系都是以左上角为原点,向右和向下形成一个x,y坐标系,表示一张图片的大小就用原点坐标加上图片的宽高即可(0,0,width,height)

2.截图原理 

opencv鼠标截取图片的原理是根据你要截图区域的左上角的坐标值和右下角的坐标值,计算得出所截取区域的width值和height值,通过opencv Rect函数实现区域框选,然后将区域提取出来放在一个新的Mat 变量中。

这里我们先讲一下手动输入框选区域的方法,定义一个Rect变量,输入框选区域的左上角坐标值,width和height

#include <iostream>
#include <opencv2/opencv.hpp>

using namespace std;
using namespace cv;

int main(void)
{
	std::cout << "start\n";
	Mat image = imread("C:\\Users\\Lenovo\\Desktop\\1.png");
	if (image.empty())
	{
		printf("无法读取到图片");
		return -1;
	}

	//手动输入框选区域
	Rect rect(470, 130, 250, 250);
	//在图片上绘制框选区域
	rectangle(image, rect, Scalar(0, 0, 255), 2, 8, 0);
	imshow("绘制输出", image);
    
	Mat dst;     //创建一个新的Mat变量来存储框选区域
	dst = image(rect);    //把框选的区域赋值给dst
	imshow("框选输出", dst);
    //有需要可以自行储存
    //imwrite("C:\\Users\\Lenovo\\Desktop\\2.png",dst);

	waitKey(0);
	destroyAllWindows();

	std::cout << "end\n";
	return 0;

}

程序实现:

 3.鼠标定点原理:

首先要实现鼠标控制,要用到鼠标的回调函数setMouseCallback(),下面是这个函数的原型和使用参数

void setMouseCallback(const String& winname, 
                        MouseCallback onMouse, 
                        void* userdata = 0);

winname 是绑定窗口

onMouse 是鼠标回调函数

userdata 是回调函数

下面是鼠标事件对应的代码和说明(这里我只列举几个常用的)

                                     EVENT                  说明
                           EVENT_MOUSEMOVE                滑动
                           EVENT_LBUTTONDOWN                左键点击
                           EVENT_RBUTTONDOWN                右键点击
                           EVENT_MBUTTONDOWN                中键点击
                           EVENT_LBUTTONUP                左键放开
                           EVENT_RBUTTONUP                右键放开
                           EVENT_MBUTTONUP                中键放开
                          EVENT_LBUTTONDBLCLK                左键双击
                          EVENT_RBUTTONDBLCLK                右键双击
                          EVENT_MBUTTONDBLCLK                中键双击
                         EVENT_FLAG_LBUTTON                左键拖拽
                        EVENT_FLAG_RBUTTON                右键拖拽
                        EVENT_FLAG_MBUTTON                中间拖拽

 接下来我们就是通过回调函数onMouse()来判断执行我们鼠标的定点功能

下面是实现代码:

#include <iostream>
#include <opencv2/opencv.hpp>


using namespace std;
using namespace cv;

Point sp(-1, -1);
Point ep(-1, -1);
Mat temp;

static void on_draw(int event, int x, int y, int flags, void* userdata) {
	Mat image = *((Mat*)userdata);
	if (event == EVENT_LBUTTONDOWN) {
		sp.x = x;
		sp.y = y;
		std::cout << "start point:" << sp << std::endl;

	}
}

void mouse_drawing_demo(Mat& image) {
	namedWindow("鼠标绘制", WINDOW_AUTOSIZE);
	setMouseCallback("鼠标绘制", on_draw, (void*)(&image));
	imshow("鼠标绘制", image);
	//temp = image.clone();
}

int main(void)
{
	std::cout << "start\n";
	Mat img = imread("C:\\Users\\Lenovo\\Desktop\\1.png");
	if (img.empty())
	{
		printf("无法读取到图片");
		return -1;
	}
	mouse_drawing_demo(img);
	waitKey(0);
	destroyAllWindows();

	std::cout << "end\n";
	return 0;

}

 执行结果: 可以看到控制台输出的坐标值,每当按下左键,我们便用一个Point记录该值便会传给控制台一个坐标值

 根据上面的鼠标定点,我们同样可以记录鼠标松开时候的坐标值,然后通过计算做两个点的width值和height。

4.鼠标框选实现

一般的款选只有左上角定点和右下角定点,看起来比较low,达不到我们平常使用的自由截图效果

从各个角度都可以截取图片,其实也很简单,在此基础上改造一下就可以了

废话就不多说了,高手都是自己研究代码

直接上代码:

#include <iostream>
#include <opencv2/opencv.hpp>


using namespace std;
using namespace cv;

Point sp(-1, -1);
Point ep(-1, -1);
Mat temp;
static void on_draw(int event, int x, int y, int flags, void* userdata) {
	Mat image = *((Mat*)userdata);
	if (event == EVENT_LBUTTONDOWN) {     //记录鼠标按下的坐标
		sp.x = x;
		sp.y = y;
		std::cout << "start point:" << sp << std::endl;			//控制台输出

	}
	else if (event == EVENT_LBUTTONUP) {			//鼠标抬起
		ep.x = x;							//记录值
		ep.y = y;
		int dx = abs(ep.x - sp.x);				//计算width	 这里为什么用abs()?
		int dy = abs(ep.y - sp.y);				// height值		如果你松开的值小于按下的值,那得到的是一个负值,框选大小没有负值(dddd)
		Rect box;
		if (dx > 0 && dy > 0) {					//判断款选
			if ((ep.x - sp.x) > 0 && (ep.y - sp.y) > 0) {			//以左上角为起点,右下角为结点框选
				box.x = sp.x;
				box.y = sp.y;
				box.width = dx;
				box.height = dy;
				cv::rectangle(image, box, Scalar(0, 0, 255), 2, 8, 0);   //绘制框选区域
			}
			else if ((ep.x - sp.x) > 0 && (ep.y - sp.y) < 0) {		//以左下角为起点,右上角为结点框选
				box.x = sp.x;
				box.y = sp.y - dy;
				box.width = dx;
				box.height = dy;
				cv::rectangle(image, box, Scalar(0, 0, 255), 2, 8, 0);
			}
			else if ((ep.x - sp.x) < 0 && (ep.y - sp.y) > 0) {			//以右上角为起点,左下角为结点框选
				box.x = sp.x - dx;
				box.y = sp.y;
				box.width = dx;
				box.height = dy;
				cv::rectangle(image, box, Scalar(0, 0, 255), 2, 8, 0);
			}
			else {												//以右下角为起点,左上角为结点框选
				{
					box.x = sp.x - dx;
					box.y = sp.y - dy;
					box.width = dx;
					box.height = dy;
					cv::rectangle(image, box, Scalar(0, 0, 255), 2, 8, 0);
				}
			}
			temp.copyTo(image);
			imshow("鼠标绘制", image);
			imshow("ROI区域", image(box));
			//图片保存功能,根据需要自行修改
			//imwrite("C:\\Users\\Lenovo\\Desktop\\3.png", image(box));    
			//ready for next drawing
			sp.x = -1;
			sp.y = -1;
		}
	}
	else if (event == EVENT_MOUSEMOVE) {			//鼠标移动 为的是实现款选区域的线看起来是动态的
		if (sp.x > 0 && sp.y > 0) {
			ep.x = x;
			ep.y = y;
			int dx = abs(ep.x - sp.x);
			int dy = abs(ep.y - sp.y);
			Rect box;
			if ((ep.x - sp.x) > 0 && (ep.y - sp.y) > 0) {
				box.x = sp.x;
				box.y = sp.y;
				box.width = dx;
				box.height = dy;
				temp.copyTo(image);		//把temp中的图片复制给image,达到一种清屏的效果	 不明白的可以自行注释这两条代码运行看看就懂了		
				cv::rectangle(image, box, Scalar(0, 0, 255), 2, 8, 0);			//清屏后立刻重新款选

			}
			else if ((ep.x - sp.x) > 0 && (ep.y - sp.y) < 0) {
				box.x = sp.x;
				box.y = sp.y - dy;
				box.width = dx;
				box.height = dy;
				temp.copyTo(image);
				cv::rectangle(image, box, Scalar(0, 0, 255), 2, 8, 0);

			}
			else if ((ep.x - sp.x) < 0 && (ep.y - sp.y) > 0) {
				box.x = sp.x - dx;
				box.y = sp.y;
				box.width = dx;
				box.height = dy;
				temp.copyTo(image);
				cv::rectangle(image, box, Scalar(0, 0, 255), 2, 8, 0);

			}
			else {
				box.x = sp.x - dx;
				box.y = sp.y - dy;
				box.width = dx;
				box.height = dy;
				temp.copyTo(image);
				cv::rectangle(image, box, Scalar(0, 0, 255), 2, 8, 0);

			}
			imshow("鼠标绘制", image);


		}
	}
}

void mouse_drawing_demo(Mat& image) {
	namedWindow("鼠标绘制", WINDOW_AUTOSIZE);
	setMouseCallback("鼠标绘制", on_draw, (void*)(&image));
	imshow("鼠标绘制", image);
	temp = image.clone();			//复制image图片
}

int main(void)
{
	std::cout << "start\n";
	Mat img = imread("C:\\Users\\Lenovo\\Desktop\\1.png");
	if (img.empty())
	{
		printf("无法读取到图片");
		return -1;
	}
	mouse_drawing_demo(img);
	waitKey(0);
	destroyAllWindows();

	std::cout << "end\n";
	return 0;

}

效果图:

 

 

 5.最后留一个小bug

截取区域越界怎么办,该怎么修改???

方法很简单,留给《聪明的读者?》自行解决

  • 4
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我了解您的问题。您可以按照以下步骤在Python中使用OpenCV实现区域截图: 1. 导入cv2库:在Python脚本的开头,输入“import cv2”命令来导入cv2库。 2. 创建窗口:使用cv2.namedWindow()方法创建一个窗口,以便在其中打开图像。 3. 加载图像:使用cv2.imread()方法从文件中加载一张图像,并将其放入刚刚创建的窗口中。 4. 设定鼠标事件回调函数:对这个窗口设置一个鼠标事件回调函数,以便在鼠标事件发生时自动触发。 5. 实现鼠标事件回调函数:在回调函数中实现鼠标事件的处理。您可以定义一个变量来记录鼠标点击事件的状态(比如起始点和结束点),并使用cv2.rectangle方法在窗口中画出一个矩形框来表示所选区域。 6. 保存截图:使用cv2.imwrite()方法将截图保存为一个图像文件。 下面是一个实现功能的简化代码示例: import cv2 def capture(event, x, y, flags, param): global drawing, ix, iy, fx, fy, screenshot if event == cv2.EVENT_LBUTTONDOWN: drawing = True ix, iy = x, y elif event == cv2.EVENT_MOUSEMOVE: if drawing == True: fx, fy = x, y cv2.rectangle(img, (ix, iy), (fx, fy), (0, 255, 0), 1) elif event == cv2.EVENT_LBUTTONUP: drawing = False fx, fy = x, y cv2.rectangle(img, (ix, iy), (fx, fy), (0, 255, 0), 1) if ix > fx: ix, fx = fx, ix if iy > fy: iy, fy = fy, iy screenshot = img[iy:fy, ix:fx] save(screenshot) def save(img): cv2.imwrite("screenshot.png", img) drawing = False ix, iy, fx, fy = -1, -1, -1, -1 img = cv2.imread('image.jpg') cv2.namedWindow('image') cv2.setMouseCallback('image', capture) while(1): cv2.imshow('image', img) if cv2.waitKey(20) & 0xFF == 27: break cv2.destroyAllWindows() 在上面的示例代码中,使用鼠标事件回调函数capture()来进行区域截图。当鼠标左键按下时,drawing变量被设置为True,并记录起始点的坐标(ix, iy)。当鼠标移动时,继续记录鼠标当前位置的坐标,并更新矩形框的位置和大小。当鼠标左键释放时,drawing变量被设置为False,记录结束点的坐标(fx, fy),并使用ix、iy、fx和fy变量来裁剪出截图区域。最后,调用save()函数将截图保存为一个文件。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值