在使用halcon的时候觉得里面的灰度处理工具很方便、灵活,之前在学opencv的时候虽然能理解阈值分割,但使用起来不方便,灰度值也是在那里随便试试,完全看不到效果。看到halcon的gen工具后自己也想用MFC实现,就花了点时间完成这个功能了。具体界面如下:
代码如下:
头文件:
#pragma once
#include "CvvImage.h"
#include
#include
#include "afxcmn.h"
using namespace std;
using namespace cv;
// ChistogramDlg 对话框
class ChistogramDlg : public CDialogEx
{
// 构造
public:
ChistogramDlg(CWnd* pParent = NULL); // 标准构造函数
// 对话框数据
enum { IDD = IDD_HISTOGRAM_DIALOG };
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支持
// 实现
protected:
HICON m_hIcon;
// 生成的消息映射函数
virtual BOOL OnInitDialog();
afx_msg void OnSysCommand(UINT nID, LPARAM lParam);
afx_msg void OnPaint();
afx_msg HCURSOR OnQueryDragIcon();
DECLARE_MESSAGE_MAP()
public:
void showImage(Mat& image, UINT ID);
afx_msg void OnBnClickedOpenpic();
afx_msg void OnBnClickedGrayprocess();
afx_msg void OnBnClickedHistogram();
int m_iMin;
int m_iMax;
CSliderCtrl m_SliderMin;
CSliderCtrl m_SliderMax;
afx_msg void OnCustomdrawMin(NMHDR *pNMHDR, LRESULT *pResult);
afx_msg void OnCustomdrawMax(NMHDR *pNMHDR, LRESULT *pResult);
};
源文件:
// histogramDlg.cpp : 实现文件
//
#include "stdafx.h"
#include "histogram.h"
#include "histogramDlg.h"
#include "afxdialogex.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#endif
Mat image;
Mat hist;
int maxCount=0; //找出灰度值的最大个数,
int grayVal; //灰度值
Mat minMaxHist;
Mat ChangeImage; //改变灰度值显示的图像区域
// 用于应用程序“关于”菜单项的 CAboutDlg 对话框
class CAboutDlg : public CDialogEx
{
public:
CAboutDlg();
// 对话框数据
enum { IDD = IDD_ABOUTBOX };
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支持
// 实现
protected:
DECLARE_MESSAGE_MAP()
};
CAboutDlg::CAboutDlg() : CDialogEx(CAboutDlg::IDD)
{
}
void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
}
BEGIN_MESSAGE_MAP(CAboutDlg, CDialogEx)
END_MESSAGE_MAP()
// ChistogramDlg 对话框
ChistogramDlg::ChistogramDlg(CWnd* pParent /*=NULL*/)
: CDialogEx(ChistogramDlg::IDD, pParent)
, m_iMin(0)
, m_iMax(255)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}
void ChistogramDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
DDX_Text(pDX, IDC_MINEDIT, m_iMin);
DDX_Text(pDX, IDC_MAXEDIT, m_iMax);
DDX_Control(pDX, IDC_MIN, m_SliderMin);
DDX_Control(pDX, IDC_MAX, m_SliderMax);
}
BEGIN_MESSAGE_MAP(ChistogramDlg, CDialogEx)
ON_WM_SYSCOMMAND()
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
ON_BN_CLICKED(IDC_OPENPIC, &ChistogramDlg::OnBnClickedOpenpic)
ON_BN_CLICKED(IDC_GRAYPROCESS, &ChistogramDlg::OnBnClickedGrayprocess)
ON_BN_CLICKED(IDC_HISTOGRAM, &ChistogramDlg::OnBnClickedHistogram)
ON_WM_HSCROLL()
ON_NOTIFY(NM_CUSTOMDRAW, IDC_MIN, &ChistogramDlg::OnCustomdrawMin)
ON_NOTIFY(NM_CUSTOMDRAW, IDC_MAX, &ChistogramDlg::OnCustomdrawMax)
END_MESSAGE_MAP()
// ChistogramDlg 消息处理程序
BOOL ChistogramDlg::OnInitDialog()
{
CDialogEx::OnInitDialog();
// 将“关于...”菜单项添加到系统菜单中。
// IDM_ABOUTBOX 必须在系统命令范围内。
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
ASSERT(IDM_ABOUTBOX < 0xF000);
CMenu* pSysMenu = GetSystemMenu(FALSE);
if (pSysMenu != NULL)
{
BOOL bNameValid;
CString strAboutMenu;
bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
ASSERT(bNameValid);
if (!strAboutMenu.IsEmpty())
{
pSysMenu->AppendMenu(MF_SEPARATOR);
pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
}
}
// 设置此对话框的图标。当应用程序主窗口不是对话框时,框架将自动
// 执行此操作
SetIcon(m_hIcon, TRUE); // 设置大图标
SetIcon(m_hIcon, FALSE); // 设置小图标
// TODO: 在此添加额外的初始化代码
UpdateData(FALSE);
m_SliderMin.SetRange(0,255);
m_SliderMax.SetRange(0,255);
m_SliderMax.SetPos(255);
m_iMin=m_SliderMin.GetPos();
m_iMax=m_SliderMax.GetPos();
return TRUE; // 除非将焦点设置到控件,否则返回 TRUE
}
void ChistogramDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
if ((nID & 0xFFF0) == IDM_ABOUTBOX)
{
CAboutDlg dlgAbout;
dlgAbout.DoModal();
}
else
{
CDialogEx::OnSysCommand(nID, lParam);
}
}
// 如果向对话框添加最小化按钮,则需要下面的代码
// 来绘制该图标。对于使用文档/视图模型的 MFC 应用程序,
// 这将由框架自动完成。
void ChistogramDlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // 用于绘制的设备上下文
SendMessage(WM_ICONERASEBKGND, reinterpret_cast(dc.GetSafeHdc()), 0);
// 使图标在工作区矩形中居中
int cxIcon = GetSystemMetrics(SM_CXICON);
int cyIcon = GetSystemMetrics(SM_CYICON);
CRect rect;
GetClientRect(&rect);
int x = (rect.Width() - cxIcon + 1) / 2;
int y = (rect.Height() - cyIcon + 1) / 2;
// 绘制图标
dc.DrawIcon(x, y, m_hIcon);
}
else
{
CDialogEx::OnPaint();
}
}
//当用户拖动最小化窗口时系统调用此函数取得光标
//显示。
HCURSOR ChistogramDlg::OnQueryDragIcon()
{
return static_cast(m_hIcon);
}
void ChistogramDlg::showImage(Mat &image, UINT ID)
{
CDC* pDC = GetDlgItem(ID)->GetDC(); //获取显示图像控件
HDC pHdc = pDC->GetSafeHdc(); //获取HDC(设备句柄)来进行绘图操作
CRect rect;
GetDlgItem(ID)->GetClientRect(&rect);
CvvImage cimg;
IplImage cpy = image;
cimg.CopyOf(&cpy);
cimg.DrawToHDC(pHdc, rect); //将图片绘制到显示控件的指定区域内
ReleaseDC(pDC);
}
void ChistogramDlg::OnBnClickedOpenpic()
{
// TODO: 在此添加控件通知处理程序代码
image= imread("test.png",1);
if (image.empty())
{
AfxMessageBox(_T("读取图片失败,请确保图片存在!"));
return;
}
showImage(image,IDC_PIC);
}
void ChistogramDlg::OnBnClickedGrayprocess()
{
// TODO: 在此添加控件通知处理程序代码
if(image.empty())
{
return;
}
if (image.channels()==3)
{
cvtColor(image,image,CV_BGR2GRAY);
}
showImage(image,IDC_GRAY);
}
void ChistogramDlg::OnBnClickedHistogram()
{
// TODO: 在此添加控件通知处理程序代码
if (image.empty())
{
return;
}
if (image.channels()==3)
{
AfxMessageBox(_T("请先进行灰度处理!"));
return;
}
//遍历灰度值
//int grayVal; //定义为global
vector vectGrayVal;
for (int i=0;i(i,j);
vectGrayVal.push_back(grayVal);
}
}
int count=0;
vector vectCount;
for (int k=0;k<256;k++)
{
for (int i=0;imaxCount)
{
maxCount=vectCount[i];
}
}
//Mat hist(maxCount,256,CV_8U); //定义为global
hist=Mat(maxCount,256,CV_8U);
minMaxHist=Mat(maxCount,256,CV_8U);
for (int k=0;k<256;k++)
{
line(hist,Point(k,maxCount),Point(k,maxCount-vectCount[k]),Scalar(0));
}
showImage(hist,IDC_HIST);
}
void ChistogramDlg::OnCustomdrawMin(NMHDR *pNMHDR, LRESULT *pResult)
{
LPNMCUSTOMDRAW pNMCD = reinterpret_cast(pNMHDR);
// TODO: 在此添加控件通知处理程序代码
if (image.empty()||hist.empty())
{
return;
}
image.copyTo(ChangeImage);
Mat minHist(maxCount,256,CV_8U);
UpdateData(TRUE);
m_iMin=m_SliderMin.GetPos();
line(minHist,Point(m_iMin,0),Point(m_iMin,maxCount),Scalar(0));
addWeighted(hist,0.5,minHist,0.5,0.0,minMaxHist);
showImage(minMaxHist,IDC_HIST);
//Mat ChangeImage; //定义为global
for (int i=0;i(i,j);
if (grayVal>m_iMin&&grayVal(i,j)=0;
}
}
}
showImage(ChangeImage,IDC_GRAY);
UpdateData(FALSE);
*pResult = 0;
}
void ChistogramDlg::OnCustomdrawMax(NMHDR *pNMHDR, LRESULT *pResult)
{
LPNMCUSTOMDRAW pNMCD = reinterpret_cast(pNMHDR);
// TODO: 在此添加控件通知处理程序代码
if (image.empty()||hist.empty())
{
return;
}
image.copyTo(ChangeImage);
UpdateData(TRUE);
m_iMax=m_SliderMax.GetPos();
Mat maxHist(maxCount,256,CV_8U);
line(maxHist,Point(m_iMax,0),Point(m_iMax,maxCount),Scalar(0));
addWeighted(hist,0.5,maxHist,0.5,0.0,minMaxHist);
showImage(minMaxHist,IDC_HIST);
image.copyTo(ChangeImage);
for (int i=0;i(i,j);
if (grayVal>m_iMin&&grayVal(i,j)=0;
}
}
}
showImage(ChangeImage,IDC_GRAY);
UpdateData(FALSE);
*pResult = 0;
}
本来分割时的滚动条是打算使用一个,上面两个滑块的,但在MFC里面没有,需要自绘,自己也没接触过,会的可以自己修改,自己也打算学一下,有相关的资料或者视频(最好)麻烦推荐给我。
另外,对于滑块的使用我知道有两种,一种是右击控件,选择类向导,在消息框中选择NM_CUSTOMDRAW,然后在各自的函数中编程(就像上面一样),或者点击后缀为DLG的属性,由于滚动条是水平的,选择WM_HSCROLL,在这个函数里同时设置两个滚动条即可。
下面是代码的连接(版本是VS2010+opencv2411的,下载后请自行修改配置):
https://download.csdn.net/download/logan_lin/10604266