MFC同时打开多个文件与多个目录

先直接上代码,给需要的人,Win7,x86,x64,VS2010下测试成功。

如果你想看看这个问题解决的过程,代码后面有详细的描述。

/********************************************************************
created:	2008/07/22
created:	22:7:2008   10:23
filename: 	SelectDialog.h
file base:	SelectDialog
file ext:	h
author:		Hojjat Bohlooli - [email protected]

purpose:	select multiple file and folders together in browse dialog
free for non commercial uses.
*********************************************************************/
#pragma once

#include <dlgs.h> // for (MULTI)FILEOPENORD

// CSelectDialog
class CSelectDialog : public CFileDialog
{
	DECLARE_DYNAMIC(CSelectDialog)

public:
	CSelectDialog(BOOL bOpenFileDialog, // TRUE for FileOpen, FALSE for FileSaveAs
		LPCTSTR lpszDefExt = NULL,
		LPCTSTR lpszFileName = NULL,
		DWORD dwFlags = OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT |
		OFN_EXPLORER & (~OFN_SHOWHELP),
		LPCTSTR lpszFilter = NULL,
		CWnd* pParentWnd = NULL);
	virtual ~CSelectDialog();

protected:
	virtual void OnInitDone();
	virtual void OnFolderChange();
	virtual BOOL OnFileNameOK();
	static LRESULT CALLBACK WindowProcNew(HWND hwnd,UINT message, WPARAM wParam, LPARAM lParam);
	DECLARE_MESSAGE_MAP()

public:
	static CString m_strCurrendDirectory;
	static CStringArray m_SelectedItemList;	/*this list includes files and folders
											are selected by user. */
	static WNDPROC m_wndProc;
};

#ifdef User
CSelectDialog sd(TRUE, NULL, NULL, OFN_ALLOWMULTISELECT | OFN_HIDEREADONLY | OFN_FILEMUSTEXIST);

if (sd.DoModal() != IDOK)
{
	return;
}

size_t nCount = sd.m_SelectedItemList.GetCount();
#endif // User

上述头文件的最后面,包含一个使用范例,接下来是cpp文件。

/********************************************************************
created:	2008/07/22
created:	22:7:2008   10:25
filename: 	SelectDialog.cpp
file base:	SelectDialog
file ext:	cpp
author:		Hojjat Bohlooli - [email protected]

purpose:	
*********************************************************************/
#include "stdafx.h"
#include "SelectDialog.h"

#pragma warning( push )
#pragma warning( disable : 4311 4312 )
// CSelectDialog
CString CSelectDialog::m_strCurrendDirectory;
CStringArray CSelectDialog::m_SelectedItemList;
WNDPROC CSelectDialog::m_wndProc = NULL;

IMPLEMENT_DYNAMIC(CSelectDialog, CFileDialog)

	CSelectDialog::CSelectDialog(BOOL bOpenFileDialog,
	LPCTSTR lpszDefExt,
	LPCTSTR lpszFileName,
	DWORD dwFlags,
	LPCTSTR lpszFilter,
	CWnd* pParentWnd)
	:CFileDialog(
	bOpenFileDialog,
	lpszDefExt,
	lpszFileName,
	dwFlags | OFN_EXPLORER | OFN_HIDEREADONLY  &(~OFN_SHOWHELP),
	lpszFilter,
	pParentWnd, 0, FALSE)
{
	dwFlags |= (OFN_EXPLORER | OFN_HIDEREADONLY &(~OFN_SHOWHELP));
};


CSelectDialog::~CSelectDialog()
{
};

BEGIN_MESSAGE_MAP(CSelectDialog, CFileDialog)
END_MESSAGE_MAP()

// CSelectDialog message handlers
BOOL CSelectDialog::OnFileNameOK()
{
	if (CFileDialog* pDlg = (CFileDialog*)CWnd::FromHandle(GetParent()->m_hWnd))
	{
		CWnd* pWnd = pDlg->GetDlgItem(lst2);	//getting list
		if (pWnd == NULL)
			return FALSE;

		m_SelectedItemList.RemoveAll();			// emptying list

		CListCtrl* wndLst1 = (CListCtrl*)(pWnd->GetDlgItem(1));

		int nSelected = wndLst1->GetSelectedCount();
		if (!nSelected)		// nothing selected -- don't retrieve list
			return FALSE;
		CString strItemText, strDirectory = m_strCurrendDirectory;
		if (strDirectory.Right(1) != _T("\\"))
			strDirectory += _T("\\");

		CString fileslist = _T("");
		pDlg->SendMessage(CDM_GETSPEC, (WPARAM)MAX_PATH,
			(LPARAM)fileslist.GetBuffer(MAX_PATH));
		fileslist.ReleaseBuffer();

		strItemText = strDirectory + fileslist;
		if(nSelected == 1 && fileslist != _T(""))
		{
			m_SelectedItemList.Add(strItemText);
			return CFileDialog::OnFileNameOK();
		}
	}
	::MessageBeep( MB_ICONQUESTION );
	return 1; //don't let the dialog to close
};

void CSelectDialog::OnFolderChange()
{
	m_strCurrendDirectory = GetFolderPath();
	CFileDialog::OnFolderChange();
};

void CSelectDialog::OnInitDone()
{
	m_strCurrendDirectory = GetFolderPath();
	CWnd* pFD = GetParent();

	HideControl(edt1);
	HideControl(cmb1);
	HideControl(stc2);

	//HideControl(cmb13);
	//HideControl(stc3);

	CRect rectCancel; pFD->GetDlgItem(IDCANCEL)->GetWindowRect(&rectCancel);
	pFD->ScreenToClient(&rectCancel);

	CRect rectOK; pFD->GetDlgItem(IDOK)->GetWindowRect(&rectOK);
	pFD->ScreenToClient(&rectOK);
	pFD->GetDlgItem(IDOK)->SetWindowPos(0,rectCancel.left - rectOK.Width() - 5, rectCancel.top, 0,0, SWP_NOZORDER | SWP_NOSIZE);

	CRect rectList2; pFD->GetDlgItem(lst1)->GetWindowRect(&rectList2);
	pFD->ScreenToClient(&rectList2);
	pFD->GetDlgItem(lst1)->SetWindowPos(0,0,0,rectList2.Width(), abs(rectList2.top - (rectCancel.top - 5)), SWP_NOMOVE | SWP_NOZORDER);

	CRect rectStatic;pFD->GetDlgItem(stc3)->GetWindowRect(&rectStatic);
	pFD->ScreenToClient(&rectStatic);
	pFD->GetDlgItem(stc3)->SetWindowPos(0,rectCancel.left - 375,rectCancel.top + 5, rectStatic.Width(), rectStatic.Height(), SWP_NOZORDER);

	CRect rectEdit1;pFD->GetDlgItem(cmb13)->GetWindowRect(&rectEdit1);
	pFD->ScreenToClient(&rectEdit1);
	pFD->GetDlgItem(cmb13)->SetWindowPos(0,rectCancel.left - 320,rectCancel.top, rectEdit1.Width() - 15, rectEdit1.Height(), SWP_NOZORDER);

	//SetControlText(stc3, _T("Item name:"));
	//SetControlText(IDOK, _T("Select"));

#ifdef _WIN64
	m_wndProc = (WNDPROC)::SetWindowLongPtr(pFD->m_hWnd, GWLP_WNDPROC, (LONG_PTR)WindowProcNew);
#else
	m_wndProc = (WNDPROC)::SetWindowLong(pFD->m_hWnd, GWL_WNDPROC, (long)WindowProcNew);
#endif // _WIN64
	pFD->CenterWindow();
};

//LRESULT CALLBACK CSelectDialog::WindowProcNew(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
#ifdef _WIN64
LONG_PTR CALLBACK CSelectDialog::WindowProcNew(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
#else
LRESULT CALLBACK CSelectDialog::WindowProcNew(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
#endif // _WIN64
{
	if (message ==  WM_COMMAND)
	{
		if (HIWORD(wParam) == BN_CLICKED)
		{
			if (LOWORD(wParam) == IDOK)
			{
				if (CFileDialog* pDlg = (CFileDialog*)CWnd::FromHandle(hwnd))
				{
					m_SelectedItemList.RemoveAll();			// emptying list
					CWnd* pWnd = pDlg->GetDlgItem(lst2);	//getting list
					if (pWnd == NULL)
						return FALSE;

					CListCtrl* wndLst1 = (CListCtrl*)(pWnd->GetDlgItem(1));

					int nSelected = wndLst1->GetSelectedCount();
					if (!nSelected)		// nothing selected -- don't retrieve list
						return FALSE;
					CString strItemText, strDirectory = m_strCurrendDirectory;
					if (strDirectory.Right(1) != _T("\\"))
						strDirectory += _T("\\");

					int nItem = wndLst1->GetNextItem(-1,LVNI_SELECTED);
					CString fileslist = _T("");
					pDlg->SendMessage(CDM_GETSPEC, (WPARAM)MAX_PATH,
						(LPARAM)fileslist.GetBuffer(MAX_PATH));
					fileslist.ReleaseBuffer();
					//////////////////   Add directory names to list
					while((nSelected--) > 0)
					{
						strItemText = wndLst1->GetItemText(nItem,0);
						strItemText = strDirectory + strItemText;
						DWORD attr = GetFileAttributes(strItemText);
						if((attr != 0xFFFFFFFF) && (attr & FILE_ATTRIBUTE_DIRECTORY))
							m_SelectedItemList.Add(strItemText);							
						nItem = wndLst1->GetNextItem(nItem, LVNI_SELECTED);
					}
					//////////////////   Add FILE names to list
					strItemText = _T("");
					nSelected = wndLst1->GetSelectedCount();
					if(nSelected > m_SelectedItemList.GetCount())
					{
						int MoreThanOnFile = fileslist.Find(_T("\""));
						if(MoreThanOnFile != -1)
						{
							for(int i=0; i<fileslist.GetLength(); i++)
								if(fileslist[i] != '\"')
								{
									strItemText.AppendFormat(_T("%c"),fileslist[i]);
									if(fileslist[i-1] == '\"' && fileslist[i] == ' ')
										strItemText.Delete(strItemText.GetLength()-1);
								}
								else if(!strItemText.IsEmpty())
								{
									m_SelectedItemList.Add((strDirectory+strItemText));
									strItemText.Empty();
								}
						}
						else
							m_SelectedItemList.Add(strDirectory+fileslist);
					}
					pDlg->EndDialog(IDOK);
					return NULL;
				} // if IDOK
			}
		} // if BN_CLICKED
	}// if WM_COMMAND
	return CallWindowProc(m_wndProc, hwnd, message, wParam, lParam);
}

#pragma warning( pop )

以下为说明:

MFC中打开多个文件并不困难,打开多个目录,却没有直接可用的代码,好在网上有个好心人在早年的时候,根据CFileDialog写了一个类SelectDialog,原文下载地址是:https://www.codeproject.com/Articles/28015/SelectDialog-A-Multiple-File-and-Folder-Select-Dia

但是存在问题需要修改,vista系统之后无法正常使用,而且该代码只适用于x86系统,也就是32位,如果想让64位系统也可以使用,则需要做一些修改。

文章支持选择文件和目录(可多选)的文件对话框CSelectDialog也包含源代码,不过是老代码,那么重点来了,这个代码要怎么修改?

在文章"支持选择文件和目录(可多选)的文件对话框CSelectDialog"的评论中,Csdn用户bjgxjob提出了修改建议:

要想在Windows Vista之后的版本调用这个对话框,需要修改构造函数去掉Vista风格,代码如下:

CSelectDialog::CSelectDialog(BOOL bOpenFileDialog,
LPCTSTR lpszDefExt,
LPCTSTR lpszFileName,
DWORD dwFlags,
LPCTSTR lpszFilter,
CWnd* pParentWnd)
:CFileDialog(
bOpenFileDialog,
lpszDefExt,
lpszFileName,
dwFlags | OFN_EXPLORER | OFN_HIDEREADONLY &amp; (~OFN_SHOWHELP),
lpszFilter,
pParentWnd, 0, FALSE)
{
dwFlags |= (OFN_EXPLORER | OFN_HIDEREADONLY &amp; (~OFN_SHOWHELP));
};


因为,Vista之后对于CFileDialog,屏蔽了一些MFC消息。

经过以上步骤,win7下x86测试正常,但是x64会有未定义的错误,而且即便简单更改未定义的变量,仍然无法正常使用。

m_wndProc = (WNDPROC)::SetWindowLong(pFD->m_hWnd, GWL_WNDPROC, (long)WindowProcNew);

经过我的调查研究,x64下回调函数发生了一些变化,为了去掉截断警告信息,我查阅到SetWindowLongPtr函数才是对应的函数,于是做了如下更改

m_wndProc = (WNDPROC)::SetWindowLongPtr(pFD->m_hWnd, GWLP_WNDPROC, (LONG_PTR)WindowProcNew);

但是问题仍然存在,这次报错比较明显,于是继续修改回调函数WindowProcNew()的返回值类型

LONG_PTR CALLBACK CSelectDialog::WindowProcNew(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)

经过以上修改和整理,总算是解决了目前存在的问题,目前测试完美。

实际上在寻找答案的过程中,我还找到了一个WTL版本的答案,也存在同样的问题,因为那份代码也是基于那篇“SelectDialog - A Multiple File and Folder Select Dialog”,所以只需要做类似的修改就可以了,这里就不贴出来了,如果需要,欢迎在文章最下方联系我。

于是这个问题就告一段落了,特此分享出来,避免需要的人再走弯路。

更多的交流,欢迎加入 暗号271888395

猜你喜欢

转载自blog.csdn.net/autumoonchina/article/details/84782856