通常情况下,我们用ShellExecute就可以实现“打开指定文件所在的目录并选中该文件”的功能,代码如下所示:
CString str = _T("/select, E:\\TestDir\\test.txt");
ShellExecute( NULL, _T("open"), _T("explorer.exe"), str, NULL, SW_SHOWNORMAL );
如上所示,使用“/select,“参数(注意:“/select”参数后面的逗号不能丢)。但用ShellExecute实现的功能是有问题的:当所在的文件夹已经打开,且选中的是其他的文件,如果此时执行ShellExecute,会将文件夹窗口置顶显示,但是不能选中目标文件。于是,查看QQ的做法,QQ是每一次都打开一个新的窗口,这样可以通过“/n, /select,“参数可以实现。但这样的处理不是最合理的,最合理的做法:如果文件夹已经打开,就不用再重新打开一次,直接置顶显示,并选中目标文件。
通过查阅相关资料,得知可以通过调用系统库shell32.dll中的非公开API函数SHOpenFolderAndSelectItems来实现上述功能,代码如下所示:
bool OpenFolderAndSelectFile( CString strFilePath ) { LPSHELLFOLDER pDesktopFolder; ::CoInitialize( NULL ); if ( FAILED( SHGetDesktopFolder( &pDesktopFolder ) ) ) { ::CoUninitialize(); return false; } LPITEMIDLIST pidl; ULONG chEaten; WCHAR wfilePath[MAX_PATH+1] = { 0 }; // IShellFolder::ParseDisplayName要传入宽字节 #ifdef _UNICODE _tcscpy( wfilePath, strFilePath ); #else MultiByteToWideChar( CP_ACP, 0, (LPCSTR)strFilePath, -1, wfilePath, MAX_PATH ); #endif LPWSTR lpWStr = wfilePath; HRESULT hr = pDesktopFolder->ParseDisplayName( NULL, 0, lpWStr, &chEaten, &pidl, NULL ); if ( FAILED( hr ) ) { pDesktopFolder->Release(); ::CoUninitialize(); return false; } LPCITEMIDLIST cpidl = pidl; bool bSucceedOrNot = false; // SHOpenFolderAndSelectItems是非公开的API函数,需要从shell32.dll获取 // 该函数只有XP及以上的系统才支持,Win2000和98是不支持的, // 如果后面要支持上述老的系统,则要添加额外的处理代码 HMODULE hShell32DLL = ::LoadLibrary( _T("shell32.dll") ); ASSERT( hShell32DLL != NULL ); if( hShell32DLL ) { typedef HRESULT (WINAPI *pSelFun)( LPCITEMIDLIST pidlFolder, UINT cidl, LPCITEMIDLIST *apidl, DWORD dwFlags ); pSelFun pFun = (pSelFun)::GetProcAddress( hShell32DLL, "SHOpenFolderAndSelectItems" ); ::FreeLibrary( hShell32DLL ); ASSERT( pFun != NULL ); if( pFun != NULL ) { // 第二个参数cidl置为0,表示是选中文件 bSucceedOrNot = SUCCEEDED(pFun(cpidl, 0, NULL, 0)); } } pDesktopFolder->Release(); // 释放pDesktopFolder ::CoUninitialize(); return bSucceedOrNot; }