www.pudn.com > MidiReader.rar > MidiReaderDlg.cpp


// MidiReaderDlg.cpp : 实现文件 
// 
 
#include "stdafx.h" 
#include "MidiReader.h" 
#include "MidiReaderDlg.h" 
 
#ifdef _DEBUG 
#define new DEBUG_NEW 
#endif 
 
string notes[]={"C","C#","D","D","E","F","F","G","G","A","A","B"}; 
 
// 用于应用程序“关于”菜单项的 CAboutDlg 对话框 
 
class CAboutDlg : public CDialog 
{ 
public: 
	CAboutDlg(); 
 
// 对话框数据 
	enum { IDD = IDD_ABOUTBOX }; 
 
	protected: 
	virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持 
 
// 实现 
protected: 
	DECLARE_MESSAGE_MAP() 
}; 
 
CAboutDlg::CAboutDlg() : CDialog(CAboutDlg::IDD) 
{ 
} 
 
void CAboutDlg::DoDataExchange(CDataExchange* pDX) 
{ 
	CDialog::DoDataExchange(pDX); 
} 
 
BEGIN_MESSAGE_MAP(CAboutDlg, CDialog) 
END_MESSAGE_MAP() 
 
 
// CMidiReaderDlg 对话框 
 
 
 
 
CMidiReaderDlg::CMidiReaderDlg(CWnd* pParent /*=NULL*/) 
	: CDialog(CMidiReaderDlg::IDD, pParent) 
{ 
	m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME); 
} 
 
void CMidiReaderDlg::DoDataExchange(CDataExchange* pDX) 
{ 
	CDialog::DoDataExchange(pDX); 
	DDX_Control(pDX, IDC_LISTINFO, m_listInfo); 
	DDX_Control(pDX, IDC_OCX1, m_player); 
} 
 
BEGIN_MESSAGE_MAP(CMidiReaderDlg, CDialog) 
	ON_WM_SYSCOMMAND() 
	ON_WM_PAINT() 
	ON_WM_QUERYDRAGICON() 
	//}}AFX_MSG_MAP 
	ON_BN_CLICKED(IDC_OPENMIDI, &CMidiReaderDlg::OnBnClickedOpenmidi) 
	ON_BN_CLICKED(IDC_PLAYMIDI, &CMidiReaderDlg::OnBnClickedPlaymidi) 
END_MESSAGE_MAP() 
 
 
// CMidiReaderDlg 消息处理程序 
 
BOOL CMidiReaderDlg::OnInitDialog() 
{ 
	CDialog::OnInitDialog(); 
 
	// 将“关于...”菜单项添加到系统菜单中。 
 
	// IDM_ABOUTBOX 必须在系统命令范围内。 
	ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX); 
	ASSERT(IDM_ABOUTBOX < 0xF000); 
 
	CMenu* pSysMenu = GetSystemMenu(FALSE); 
	if (pSysMenu != NULL) 
	{ 
		CString strAboutMenu; 
		strAboutMenu.LoadString(IDS_ABOUTBOX); 
		if (!strAboutMenu.IsEmpty()) 
		{ 
			pSysMenu->AppendMenu(MF_SEPARATOR); 
			pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu); 
		} 
	} 
 
	// 设置此对话框的图标。当应用程序主窗口不是对话框时,框架将自动 
	//  执行此操作 
	SetIcon(m_hIcon, TRUE);			// 设置大图标 
	SetIcon(m_hIcon, FALSE);		// 设置小图标 
 
	// TODO: 在此添加额外的初始化代码 
 
	return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE 
} 
 
void CMidiReaderDlg::OnSysCommand(UINT nID, LPARAM lParam) 
{ 
	if ((nID & 0xFFF0) == IDM_ABOUTBOX) 
	{ 
		CAboutDlg dlgAbout; 
		dlgAbout.DoModal(); 
	} 
	else 
	{ 
		CDialog::OnSysCommand(nID, lParam); 
	} 
} 
 
// 如果向对话框添加最小化按钮,则需要下面的代码 
//  来绘制该图标。对于使用文档/视图模型的 MFC 应用程序, 
//  这将由框架自动完成。 
 
void CMidiReaderDlg::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 
	{ 
		CDialog::OnPaint(); 
	} 
} 
 
//当用户拖动最小化窗口时系统调用此函数取得光标显示。 
// 
HCURSOR CMidiReaderDlg::OnQueryDragIcon() 
{ 
	return static_cast(m_hIcon); 
} 
 
 
void CMidiReaderDlg::OnBnClickedOpenmidi()         //打开文件夹中的midi文件  并分析其中数据 
{ 
	// TODO: 在此添加控件通知处理程序代码 
	m_listInfo.ResetContent(); 
	static char szFilter[]="Video Files||"; 
	CFileDialog FileDlg(TRUE,NULL,NULL,OFN_HIDEREADONLY,szFilter);//此处为TRUE时表示是一个打开文件的对话框		 
	if(FileDlg.DoModal()==IDOK) 
	{ 
		m_SourceFile = FileDlg.GetPathName(); 
	} 
 
//以二进制格式打开midi文件 并且读入byte数据 
	FILE *fp=0; 
	unsigned char ch; 
	int count=0;	 
	char *temp=new char[100]; 
 
	vector midi_data; 
	midi_data.reserve(40000); 
	if((fp=fopen((LPCSTR)m_SourceFile,"rb"))!=NULL) 
	{ 
		ch=fgetc(fp); 
		while(!feof(fp)) 
		{ 
			midi_data.push_back(ch); 
			sprintf(temp,"%x",ch,ch); 
 
			if(++count==14) 
			{ 
				count=0; 
 
			} 
			ch=fgetc(fp); 
		} 
	} 
 
 
	fclose(fp); 
 
//分析midi数据 
	if(midi_data[0]=='M'&&midi_data[1]=='T'&&midi_data[2]=='h'&&midi_data[3]=='d') 
	{ 
		sprintf(temp,"该文件是midi文件"); 
		m_listInfo.AddString(temp); 
 
	} 
	else 
	{ 
		sprintf(temp,"该文件不是midi文件"); 
		m_listInfo.AddString(temp); 
		delete temp; 
		return; 
	} 
 
	switch(midi_data[9]){ 
		case 0: 
			sprintf(temp,"文件格式:单轨"); 
			m_listInfo.AddString(temp); 
			break; 
		case 1: 
			sprintf(temp,"文件格式:多轨,同步"); 
			m_listInfo.AddString(temp); 
			break; 
		case 2: 
			sprintf(temp,"文件格式:多轨,异步"); 
			m_listInfo.AddString(temp); 
 
	} 
 
	int tracknum=midi_data[10]*16+midi_data[11]; 
	sprintf(temp,"文件音轨数:%d",tracknum); 
	m_listInfo.AddString(temp); 
 
 
 
	int rhythmnum=midi_data[12]*16+midi_data[13]; 
	sprintf(temp,"每个4分音符delta-time节奏数:%d",rhythmnum); 
	m_listInfo.AddString(temp); 
	m_listInfo.AddString(""); 
 
 
	sprintf(temp,"该文件的音符数据:(格式)八度音阶&音符号"); 
	m_listInfo.AddString(temp); 
 
 
//具体分析midi文件的音符数据 
	vector note_str; 
	note_str.reserve(10); 
	count=0; 
	for(vector::const_iterator i=midi_data.begin();i!=midi_data.end();i++) 
	{ 
		unsigned char tch=*i; 
		int key=tch>>4; 
		if(key==9)                    //the time of playing the note 
		{ 
			int note=static_cast(*(++i)); 
			while(note<=127) 
			{	 
				sprintf(temp,"%d&%s  ",note/12,notes[(note%12)].c_str()); 
				note_str.push_back(temp); 
				count++; 
				if(count==8) 
				{ 
					sprintf(temp,"%s%s%s%s%s%s%s%s",note_str[0].c_str(),note_str[1].c_str(),note_str[2].c_str(),note_str[3].c_str(), 
						note_str[4].c_str(),note_str[5].c_str(),note_str[6].c_str(),note_str[7].c_str()); 
					m_listInfo.AddString(temp); 
					note_str.clear(); 
					count=0; 
				} 
 
				constant_type::type space(constant(string(" "))); 
 
				if(i+1==midi_data.end()||i+2==midi_data.end()) 
				{ 
					string f_str; 
					for_each(note_str.begin(),note_str.end(),var(f_str)+=_1); 
					m_listInfo.AddString(f_str.c_str()); 
					delete temp; 
					return; 
				} 
				i+=2; 
				note=static_cast(*i); 
			} 
			i--; 
		} 
		else 
			continue; 
 
	} 
	string f_str; 
	for_each(note_str.begin(),note_str.end(),var(f_str)+=_1); 
	m_listInfo.AddString(f_str.c_str()); 
	delete temp; 
} 
 
void CMidiReaderDlg::OnBnClickedPlaymidi()           //利用media player播放midi文件 
{ 
	// TODO: 在此添加控件通知处理程序代码 
		m_player.put_URL((LPCSTR) m_SourceFile); 
}