Opencv实现拼图板游戏

 更新时间:2020年03月23日 11:28:45   作者:-牧野-  
这篇文章主要为大家详细介绍了Opencv实现拼图板小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

80后可能还对儿时玩过的一种经典木质的拼图板游戏记忆犹新,一般是一种4*4或5*5规格的手持活动板,通过挪动每个小板子的位置,拼出来板子上完整的图像,那时候还没有网吧,手机也还是大哥大的天下,所以这也可以算得上是最早的“手游”了吧……

简单的就是经典的,现在的Windows 7小工具里还保留了这个小游戏,当然你可能从来没有留意过~,就是下边的这个:

可以在控制面板->外观->桌面小工具里调出来。

这里准备用opencv里的模板匹配,通过鼠标响应事件来实现这个小游戏。

首先第一步是对图像按照传入的行列参数分割,并把分割出来的行*列个个数的子图像在另一空白图像中显示出来:

for(int i=0;i<rows;i++)
 {
 for(int j=0;j<cols;j++)
 {
 Mat SourceRoi=Sourceimage(Rect(j*Roicols,i*Roirows,Roicols-1,Roirows-1));
 arraryimage.push_back(SourceRoi);
 
 }
 }

rows和cols分别是用户定义的行列数,arraryimage是定义的 vector<Mat>类型的向量。
分割完之后需要把这些子图像随机的显示在另一空白图像中,这里写了一个生成指定区间里的不重复的随机数来实现:

//*******************************************************************//
//随机调换所有的子图像序列的位置,用于在 Splite image中显示
//*******************************************************************//
void Randarrary( vector<Mat>& vectorMat)
{
 for(int i=0;i<vectorMat.size();i++)
 {
 srand(int(time(0)));
 int a=rand()%(vectorMat.size()-i)+i;
 swap(vectorMat[i],vectorMat[a]);
 }
 
}

C++中使用rand()生成随机数记得先定义种子,不然系统会默认种子为1,这样每次生成的随机序列都是一样的,第一个随机数永远是41,关于rand()以后再说一说。
每生成一个随机数,就把该随机数作下标的向量元素跟第一个元素对换,实现生成不重复的随机数。

所有分割出来的子图像按随机顺序组成了“Splite image”图像后,通过鼠标单击事件响应函数,定位到鼠标单击点坐在的子图像,并把该子图像用模板匹配方法在原图像中定位出位置,最后合成到目标图像“Jigsaw image”

//*******************************************************************//
//鼠标回调函数,用于获取需要查找的子图像在原图像中的位置,并在叠加显示在目标图像中
//*******************************************************************//
void OnMouseAction(int event,int x,int y,int flags,void *ustc)
{
 if(event==CV_EVENT_LBUTTONDOWN)
 {
 Mat RoiSpilte,RoiSource;
 int rows=(y/Roirows)*Roirows;
 int clos=(x/Roicols)*Roicols;
 
 
 RoiSpilte=Spilteimage(Rect(clos,rows,Roicols,Roirows));
 imshow("Slice",RoiSpilte);
 
 Mat image=Mat::zeros(Sourceimage.rows-Roirows,Sourceimage.cols-Roicols,CV_32FC1);
 matchTemplate(Sourceimage,RoiSpilte,image,1);
 normalize(image,image,0,1,NORM_MINMAX);
 
 double minV=0;
 double maxV=0;
 Point minP,maxP;
 
 minMaxLoc(image,&minV,&maxV,&minP,&maxP);
 
 //Mat ROIS=Sourceimage(Rect(maxP.x,maxP.y,Roicols,Roirows));
 Mat ROIDst=Dstimage(Rect(minP.x,minP.y,Roicols,Roirows));
 addWeighted(ROIDst,0,RoiSpilte,1,0,ROIDst,-1);
 imshow("Jigsaw image",Dstimage);
 
 }

原图像:

行列分割后的图像,子图像位置随机分布:

单击Splite image图像中的子图像,叠加该子图像到目标图像上,子图像位置通过模板匹配方法在原图像中定位:

完成后效果:

为了清楚显示边界,每个子图像在行列上都减了一个像素,所以上图可见黑色线条。

完整程序:

#include "core/core.hpp"
#include "highgui/highgui.hpp"
#include "imgproc/imgproc.hpp"
#include <time.h>
 
using namespace cv;
 
Mat Sourceimage,Spilteimage,Rebuildimage,Dstimage;
int rows,cols;
int Roirows,Roicols;
vector<Mat>arraryimage;
void Randarrary( vector<Mat> &vectorMat); //随机排列子图像序列函数
static int vectornumber=0;
void OnMouseAction(int event,int x,int y,int flags,void *ustc); //鼠标回调事件函数
 
int main(int argc,char*argv[])
{
 Sourceimage=imread(argv[1]);
 imshow("Source image",Sourceimage);
 rows=atoi(argv[2]);
 cols=atoi(argv[3]);
 Roirows=Sourceimage.rows/rows;
 Roicols=Sourceimage.cols/cols;
 Spilteimage=Mat::zeros(Sourceimage.rows,Sourceimage.cols,Sourceimage.type());
 Dstimage=Mat::zeros(Sourceimage.rows,Sourceimage.cols,Sourceimage.type());
 for(int i=0;i<rows;i++)
 {
 for(int j=0;j<cols;j++)
 {
 Mat SourceRoi=Sourceimage(Rect(j*Roicols,i*Roirows,Roicols-1,Roirows-1));
 arraryimage.push_back(SourceRoi);
 }
 }
 // 随机函数
 Randarrary( arraryimage);
 
 for(int i=0;i<rows;i++)
 {
 for(int j=0;j<cols;j++)
 {
 Mat SpilterRoi=Spilteimage(Rect(j*Roicols,i*Roirows,Roicols-1,Roirows-1));
 addWeighted(SpilterRoi,0,arraryimage[vectornumber],1,0,SpilterRoi);
 vectornumber++;
 imshow("Splite image",Spilteimage);
 waitKey(150);
 }
 }
 setMouseCallback("Splite image",OnMouseAction);
 waitKey();
 
}
 
//*******************************************************************//
//随机调换所有的子图像序列的位置,用于在 Splite image中显示
//*******************************************************************//
void Randarrary( vector<Mat>& vectorMat)
{
 for(int i=0;i<vectorMat.size();i++)
 {
 srand(int(time(0)));
 int a=rand()%(vectorMat.size()-i)+i;
 swap(vectorMat[i],vectorMat[a]);
 }
 
}
 
//*******************************************************************//
//鼠标回调函数,用于获取需要查找的子图像在原图像中的位置,并在叠加显示在目标图像中
//*******************************************************************//
void OnMouseAction(int event,int x,int y,int flags,void *ustc)
{
 if(event==CV_EVENT_LBUTTONDOWN)
 {
 Mat RoiSpilte,RoiSource;
 int rows=(y/Roirows)*Roirows;
 int clos=(x/Roicols)*Roicols;
 
 RoiSpilte=Spilteimage(Rect(clos,rows,Roicols,Roirows));
 imshow("Slice",RoiSpilte);
 
 Mat image=Mat::zeros(Sourceimage.rows-Roirows,Sourceimage.cols-Roicols,CV_32FC1);
 matchTemplate(Sourceimage,RoiSpilte,image,1);
 normalize(image,image,0,1,NORM_MINMAX);
 
 double minV=0;
 double maxV=0;
 Point minP,maxP;
 
 minMaxLoc(image,&minV,&maxV,&minP,&maxP);
 
 Mat ROIDst=Dstimage(Rect(minP.x,minP.y,Roicols,Roirows));
 addWeighted(ROIDst,0,RoiSpilte,1,0,ROIDst,-1);
 imshow("Jigsaw image",Dstimage);
 }
}

资源文件和Code也可以在点击这里 拼图板小游戏

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

相关文章

  • Qt5.9实现简单复合图形

    Qt5.9实现简单复合图形

    这篇文章主要为大家详细介绍了Qt5.9实现简单复合图形,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-07-07
  • C++ TinyXML解析案例详解

    C++ TinyXML解析案例详解

    这篇文章主要介绍了C++ TinyXML解析案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • C++中BitBlt的使用方法详解

    C++中BitBlt的使用方法详解

    这篇文章主要介绍了C++中BitBlt的使用方法详解的相关资料,希望通过本文能帮助到大家,需要的朋友可以参考下
    2017-09-09
  • C++示例讲解观察者设计模式

    C++示例讲解观察者设计模式

    观察者模式是极其重要的一个设计模式,也是我几年开发过程中使用最多的设计模式,本文首先概述观察者模式的基本概念和Demo实现,接着是观察者模式在C++中的应用,最后是对观察者模式的应用场景和优缺点进行总结
    2022-12-12
  • 详解C++ sizeof(上)

    详解C++ sizeof(上)

    这篇文章主要介绍了C++ sizeof的相关资料,帮助大家更好的理解和学习c++,感兴趣的朋友可以了解下
    2020-08-08
  • C++ Boost TypeTraits库使用详解

    C++ Boost TypeTraits库使用详解

    Boost是为C++语言标准库提供扩展的一些C++程序库的总称。Boost库是一个可移植、提供源代码的C++库,作为标准库的后备,是C++标准化进程的开发引擎之一,是为C++语言标准库提供扩展的一些C++程序库的总称
    2022-11-11
  • C++教程之array数组使用示例详解

    C++教程之array数组使用示例详解

    这篇文章主要为大家介绍了C++教程之array数组使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • C++ static的作用解读

    C++ static的作用解读

    这篇文章主要介绍了C++ static的作用及说明,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02
  • 深入uCOS中全局变量的使用详解

    深入uCOS中全局变量的使用详解

    本篇文章是对uCOS中全局变量的使用进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++中priority_queue的使用与模拟实现

    C++中priority_queue的使用与模拟实现

    本文主要介绍了C++中priority_queue的使用与模拟实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-02-02

最新评论