opencv模拟景深效果

简介

  在htc的相机应用中发现了一个叫景深的拍照场景,然后练习着用opencv来模拟实现。

hcl景深场景效果

                                                   

本例实现效果

                                                   

实现过程

  要实现本例,需要用到两方面知识:1、图像的高斯模糊。 2.在图像中去出感兴趣的圆形区域。

高斯模糊

  用的是opencv函数:cvSmooth。该函数参数具体使用,看opencv官网的api介绍吧:http://docs.opencv.org/modules/refman.html
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <math.h>
#include <string.h>
#include <opencv/cv.h>
#include <stdio.h>

using namespace cv;

int width, height;
char pic_name[20];
Mat mat1;

void mySmooth(Mat mat1, Mat mat2, int flag, int width, int height){
    IplImage pI1 = mat1;
    IplImage pI2 = mat2;

    cvSmooth(&pI1, &pI2, flag, width, height);
}

int main(int argc,char *argv[]){
    memcpy(pic_name,argv[1], sizeof(argv[1]));
    mat1 = imread(pic_name, 1);

    imshow("1", mat1);
    mySmooth(mat1, mat1,CV_GAUSSIAN, 23, 23);
    imshow("2", mat1);
    cv::waitKey(0);
    return 0;
}
  显示效果如下:
                                

圆形兴趣区域

  在opencv的ROI中,最常使用的在图片中截取出感兴趣区域方式如下:
cv::Mat imageROI;
imageROI = img(cv::Rect(40,40,40,40));
  该代码表示,在图像img的坐标(40,40)位置,取出一个width:40,height:40的矩形复制到imageROI中。
但是如果我们想复制取出的区域为圆形的话,就不能用这个办法。需要使用如下方式:
/**************取出圆形感兴趣区域**********/
    mat1 = imread(pic_name, 1);
    src = mat1;
    res = cvCreateImage(cvGetSize(&src), 8, 3);
    roi = cvCreateImage(cvGetSize(&src), 8, 1);

    cvZero(roi);
    cvZero(res);
    cvCircle(roi,cvPoint(50, 50), 30,CV_RGB(255, 255, 255),-1, 8, 0);
    cvAnd(&src, &src, res, roi);
    /******************************************/
  首先新建两个和原图像src相同大小的Mat:res、roi,然后将res和roi内容都清空,接着在roi中坐标(50,50)为圆形,30为半径画一个圆。
接着调用cvAnd函数中,将roi作为掩码输入,就在res中复制得到了src对应在roi画圆区域的图像内容。
  具体的效果演示如下:
                                       

本例代码

  本例使用的代码主要就是如上知识,另外还有鼠标拖动来控制图像景深圆圈的大小和开始位置。
具体代码如下:
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <math.h>
#include <string.h>
#include <opencv/cv.h>
#include <stdio.h>

using namespace cv;

int width, height;
int pic_info[3];
char pic_name[20];
Mat mat1;
IplImage src, *res, *roi;

void mySmooth(Mat mat1, Mat mat2, int flag, int width, int height){
	IplImage pI1 = mat1;
	IplImage pI2 = mat2;

	cvSmooth(&pI1, &pI2, flag, width, height);
}

void mypic_merge(IplImage* src, IplImage* res, IplImage* dst){
	CvScalar s;
	int height = src->height;
	int width = src->width;
	int i, j;

	for(i=0;i<height;i++){
		for(j=0;j<width;j++){
			s = cvGet2D(res, i, j);
			if((s.val[0] == 0) && (s.val[1] == 0) && (s.val[2] == 0)){
				s = cvGet2D(src, i, j);
			}
			cvSet2D(dst,i,j,s);
		}
	}
}

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

	if(event == CV_EVENT_LBUTTONDOWN){
		pic_info[0] = x;   /*width1*/
		pic_info[1] = y;   /*height1*/
		pic_info[2] = 0;   /*width2*/
		mat1=cv::imread(pic_name,1);
		cv::imshow("1",mat1);
	}
	if(flags == CV_EVENT_FLAG_LBUTTON){
		x = abs(x - pic_info[0]);
		y = abs(y - pic_info[1]);
		pic_info[2] = (int)sqrt((x*x + y*y));
		mat1=cv::imread(pic_name,1);
		src = mat1;

		res = cvCreateImage(cvGetSize(&src), 8, 3);
		roi = cvCreateImage(cvGetSize(&src), 8, 1);

/**************取出圆形感兴趣区域**********/
		cvZero(roi);
		cvZero(res);
		cvCircle(roi,cvPoint(pic_info[0], pic_info[1]), pic_info[2],CV_RGB(255, 255, 255),-1, 8, 0);
		cvAnd(&src, &src, res, roi);
/******************************************/
		mySmooth(mat1, mat1,CV_GAUSSIAN, 23, 23);
		mypic_merge(&src, res, res);

		cvNamedWindow("2", 1);
		cvShowImage("2", res);
	}
	printf("circle:%d\n", pic_info[2]);
} 

int main(int argc,char *argv[]){
	memcpy(pic_name,argv[1], sizeof(argv[1]));
	mat1 = imread(pic_name, 1);
	width = mat1.rows;
	height = mat1.cols;

	imshow("1", mat1);
	cvSetMouseCallback("1", on_mouse, NULL);
	cv::waitKey(0);
	return 0;
}
  如上,就能实现在图像窗口"1"中,通过鼠标点击和拖拽,控制景深圆圈变化。
时间: 2024-12-21 15:34:31

opencv模拟景深效果的相关文章

wpf 模拟3D效果(和手机浏览图片效果相似)(附源码)

原文 wpf 模拟3D效果(和手机浏览图片效果相似)(附源码) pf的3D是一个很有意思的东西,类似于ps的效果,类似于电影动画的效果,因为动画的效果,(对于3D基础的摄像机,光源,之类不介绍,对于依赖属性也不介绍.),个人认为,依赖属性这个东西,有百分之五十是为了3D而存在.(自己写的类似于demo的东西)先上图,无图无真相这是demo的整个效果图,可以用鼠标移动,触摸屏也可以手指滑动,图片会移动,然后移动结束,会有一个回弹的判断. <Window x:Class="_3Dshow.Wi

[ZZ] 景深效果(Depth of Field) , Pass1 将场景渲染到一个RenderTarget,做为清晰版, Pass2: BluredRT , Pass3: WDepth = Depth / Far_Z_Clip

http://blog.csdn.net/xoyojank/article/details/1883520 什么是景深效果? 景深效果,简称DOF,在人眼跟光学摄像设备上很常见.如下图: 简单地来说,就是近处跟远处的景物模糊,而焦点附近的物体则很清晰.至于为什么会产生这样的效果,我就懒得说了:p 那么怎么来实现这种效果呢? 看图: 从摄像机开始,按距离分成三部分: 近距离模糊,焦点范围(清晰),远距离模糊 渲染的时候按深度(即距离)进行判断,在焦点范围内则是清晰的,否则就进行模糊处理. 整个过程

js模拟placeholder效果1

由于有的浏览器不支持placeholder,所以写了一个jquery插件来模拟placeholder效果. 使用方法:$("#IDName").placeholder(); 话不多说,直接上代码~~~ (function ($) {     $.fn.placeholder = function (options) {         var defaults = {             pColor: "#BEBEBE",             pActiv

360浏览器搜索框下拉选择图片js模拟select效果

最近360浏览器网址导航的主页增加了一个下拉选择图片搜索的功能,也就是用js模拟出了select的效果,今天在单位闲了无事干,就把空上功能给摸索出来了,虽然做的不是太完善,但对要求不高的用户来说,已经可以了,而且也可以为学习Js的朋友提供参考. <html> <head> <meta http-equiv="content-type" content="text/html; charset=utf-8"> <style>

强大的CSS:模拟下雪效果动画制作教程

下雪效果只是一类效果的名称,可以是红包雨等一些自由落体的运动效果,本文就是用纯css模拟下雪的效果,更多效果大家可以自行发挥. 1.前言 由于公司产品的活动,需要模拟类似下雪的效果.浏览器实现动画无非css3和canvas(还有gif),对比下css3和canvas的优缺点: 动画自由度:canvas胜: 复杂度:canvas胜: 兼容性:canvas胜: 性能:css3胜(requestAnimationFrame和硬件加速). 由于对于性能有一定的要求,canvas对比css3会有更多的计算

CSS:模拟下雪效果动画制作教程

下雪效果只是一类效果的名称,可以是红包雨等一些自由落体的运动效果,本文就是用纯css模拟下雪的效果,更多效果大家可以自行发挥. 1.前言 由于公司产品的活动,需要模拟类似下雪的效果.浏览器实现动画无非css3和canvas(还有gif),对比下css3和canvas的优缺点: 动画自由度:canvas胜: 复杂度:canvas胜: 兼容性:canvas胜: 性能:css3胜(requestAnimationFrame和硬件加速). 由于对于性能有一定的要求,canvas对比css3会有更多的计算

opencv模拟button

简介 在opencv中并没有专门的button控件,一般可以使用trackbar,设置状态为0.1,来模拟控制触发.不过总感觉这方法有点二,因此尝试了下在opencv中模拟 键值使用. 同时在实现过程中,使用到了资料:http://www.csdn123.com/html/mycsdn20140110/84/8403a669ceed8e896d5c3ebc45b483b4.html ,非常感谢. 实现流程 具体代码 void ButtonShow(void){ int i; for(i=0; i

android开发步步为营之69:Activity通过设置Theme模拟对话框效果

对话框应该说是手机app中最常见的和用户进行交互的方式了,之前也总结过一下,主要通过popupwindow,dialog,windowmanger来实现,本文详细介绍一下如何通过Activity实现弹出对话框效果. 来先看下效果,有个感性的认识. 中间那个提示其实是一个activity,好的,下面开始一步步实现这个神奇的效果.   第一步:设计对话框页面activity_simulate_dialog.xml <span style="font-size:14px;">&l

OpenCV 实现哈哈镜效果

代码,有参考别人的代码 // haha_mirror.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> #include "cv.h" #include "highgui.h" #include "math.h" #include "opencv2/core/core.hpp" #pragma comment(lib,&