OpenCV实现马赛克功能

 更新时间:2018年01月27日 16:24:55   作者:__輝  
这篇文章主要为大家详细介绍了OpenCV实现马赛克功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了OpenCV实现马赛克功能的具体代码,供大家参考,具体内容如下

实现用按下鼠标左键拖动时,在鼠标经过的路径上打上马赛克。

马赛克的原理是将图像中选中区域的像素用这个选中区域中的某一像素覆盖。

为了不让鼠标重复经过图像中同一个的时候,选取不一样的像素,该程序将在输入图片的时候,就实现了全图的马赛克效果。而当鼠标划过的时候,程序只是将实现马赛克的图片的指定位置复制到显示的图像中。

效果类似于QQ截图中的马赛克。

#include <opencv2\core\core.hpp> 
#include <opencv2\highgui\highgui.hpp> 
#include <opencv2\imgproc\imgproc.hpp> 
#include <iostream> 
 
using namespace cv; 
using namespace std; 
 
Mat inputImage; 
Mat inputImage_mosaic; 
Mat inputImage_clone; 
 
//马赛克的大小 
int neightbourhood = 20; 
 
//记录鼠标的状态,0为鼠标左键未按下或弹起,1为鼠标左键按下 
int mouseStatus = 0; 
 
void onMouse(int events, int x, int y, int flag, void* ustg); 
 
//创建马赛克图片 
void createMosaicImage(Mat inputMat, Mat& outputMat, int size); 
 
//设置马赛克区域 
void setMosaic(Mat& inputMat, Rect rect); 
 
int main(void){ 
 inputImage = imread("test2.jpg"); 
 inputImage_clone = inputImage.clone(); 
 createMosaicImage(inputImage, inputImage_mosaic, neightbourhood); 
 
 namedWindow("showImage", 0); 
 setMouseCallback("showImage", onMouse); 
 
 waitKey(); 
 return 0; 
} 
 
void createMosaicImage(Mat inputMat, Mat& outputMat, int size){ 
 RNG rng; 
 int height = inputMat.rows; 
 int width = inputMat.cols; 
 Mat padding; 
 Mat tempMat; 
 
 //为了方便后面的计算,将输入的图像大小扩充到宽高都是size的倍数 
 copyMakeBorder(inputMat, padding, 0, size - inputMat.rows % size, 0, size - inputMat.cols % size, BORDER_REPLICATE); 
 tempMat = padding.clone(); 
 
 for (int row = 0; row < padding.rows; row += size){ 
 for (int col = 0; col < padding.cols; col += size){ 
  int rand_x = rng.uniform(0, size); 
  int rand_y = rng.uniform(0, size); 
  Rect rect = Rect(col, row, size, size); 
  Mat roi = tempMat(rect); 
  Scalar color = Scalar(padding.at<Vec3b>(row + rand_y, col + rand_x)[0], \ 
  padding.at<Vec3b>(row + rand_y, col + rand_x)[1], \ 
  padding.at<Vec3b>(row + rand_y, col + rand_x)[2]); 
  roi.setTo(color); 
 } 
 } 
 outputMat = tempMat(Rect(0, 0, width, height)).clone(); 
} 
 
void setMosaic(Mat& inputMat, Rect rect){ 
 Mat roi = inputMat(rect); 
 Mat tempRoi = inputImage_mosaic(rect); 
 tempRoi.copyTo(roi); 
} 
 
void onMouse(int events, int x, int y, int flag, void* ustg){ 
 
 //当鼠标移除图片区域的时候,不做操作 
 if (x < 0 || x > inputImage.cols || y < 0 || y > inputImage.rows){ 
 return; 
 } 
 
 //马赛克块的位置信息 
 int x_left, x_right, y_top, y_bottom; 
 x - neightbourhood <= 0 ? x_left = 0 : x_left = x - neightbourhood; 
 x + neightbourhood > inputImage.cols ? x_right = inputImage.cols: x_right = x + neightbourhood; 
 y - neightbourhood <= 0 ? y_top = 0 : y_top = y - neightbourhood; 
 y + neightbourhood > inputImage.rows ? y_bottom = inputImage.rows: y_bottom = y + neightbourhood; 
 
 if (events == CV_EVENT_LBUTTONDOWN){ 
 mouseStatus = 1; 
 setMosaic(inputImage_clone, Rect(x_left, y_top, x_right - x_left, y_bottom - y_top)); 
 } 
 else if (events == CV_EVENT_MOUSEMOVE){ 
 if (mouseStatus == 1){ 
  setMosaic(inputImage_clone, Rect(x_left, y_top, x_right - x_left, y_bottom - y_top)); 
 } 
 else{ 
  //nothing 
 } 
 } 
 else if (events == CV_EVENT_LBUTTONUP){ 
 mouseStatus = 0; 
 } 
 else { 
 //cout << "nothing" << endl; 
 } 
 imshow("showImage", inputImage_clone); 
} 

效果图

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C语言中条件判断的正确使用姿势

    C语言中条件判断的正确使用姿势

    在C语言中,有三种条件判断结构:if语句、if-else语句和switch语句,这篇文章主要来和大家讲解一下它们的正确使用姿势,需要的可以参考一下
    2023-05-05
  • C语言 选择排序算法详解及实现代码

    C语言 选择排序算法详解及实现代码

    本文主要介绍C语言 选择排序算法,这里对排序算法做了详细说明,并附代码示例,有需要的小伙伴可以参考下
    2016-08-08
  • C++归并排序算法详解

    C++归并排序算法详解

    大家好,本篇文章主要讲的是C++归并排序算法详解,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2022-01-01
  • 用C++编写扩展node.js(node-ffi版)

    用C++编写扩展node.js(node-ffi版)

    今天小编就为大家分享一篇关于用C++编写扩展node.js(node-ffi版),小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • C语言中的sscanf()函数使用详解

    C语言中的sscanf()函数使用详解

    这篇文章主要介绍了C语言中的sscanf()函数使用详解,文中附加了一道相关的ACM题目进行补充巩固,需要的朋友可以参考下
    2015-08-08
  • C语言实现牛顿迭代法解方程详解

    C语言实现牛顿迭代法解方程详解

    这篇文章主要介绍了C语言实现牛顿迭代法解方程详解的相关资料,需要的朋友可以参考下
    2017-03-03
  • 成员初始化列表与构造函数体中的区别详细解析

    成员初始化列表与构造函数体中的区别详细解析

    无论是在构造函数初始化列表中初始化成员,还是在构造函数体中对它们赋值,最终结果是相同的。不同之处在于,使用构造函数初始化列表的版本初始化数据成员,没有定义初始化列表的构造函数版本在构造函数体中对数据成员赋值
    2013-09-09
  • C语言如何实现循环输入

    C语言如何实现循环输入

    这篇文章主要介绍了C语言如何实现循环输入问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02
  • C语言结构体指针的具体使用

    C语言结构体指针的具体使用

    结构体指针是一种非常有用的数据类型,它可以让我们更方便地操作结构体,本文主要介绍了C语言结构体指针的具体使用,非常具有实用价值,需要的朋友可以参考下
    2023-05-05
  • C语言数组实现三子棋应用实例

    C语言数组实现三子棋应用实例

    这篇文章主要为大家详细介绍了C语言数组实现三子棋应用实例,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01

最新评论