2023年12月11日发(作者:)

VC一点通:实现文件夹的缩略图显示

本示例演示了列表控件的虚列表和自画功能,也演示了一些系统外壳的函数和接口的使用方法。

单击这里下载本文的代码。

预备性阅读

在阅读本文之前,建议先对列表视图控件和系统外壳有一个基本的了解。建议阅读以下SDK文章

ShellFAQ

List-ViewControlsOverview

UsingList-ViewControls

CustomizingaControl'sAppearanceUsingCustomDraw

创建应用程序

使用MFC应用程序向导创建一个SDI应用程序,在最后一步选择视图的基类为CListView。创建完成之后,在资源中去掉保存、编辑和打印等功能的菜单和工具栏按钮(因为这些功能没有实现)。

虚列表的创建

本文采用虚列表技术,使得显示信息是在第一次显示的时候才被获取。为了创建虚列表,在创建之前需要指定列表的风格

BOOLCPicViewView::PreCreateWindow(CREATESTRUCT&cs)

{

&=~LVS_TYPEMASK;

|=LVS_ICON|LVS_OWNERDATA;

returnCListView::PreCreateWindow(cs);

} 同时,因为列表项的Overlay图标也是被动态获取的,所以需要设置动态Overlay图标

voidCPicViewView::OnInitialUpdate()

{

CListView::OnInitialUpdate();

GetListCtrl().SetCallbackMask(LVIS_OVERLAYMASK);

}

缓存显示信息

在列表需要显示一个范围的项目之前,列表会发送LVN_ODCACHEHINT通知,应用程序可以捕获这个消息来缓存部分列表的显示信息,以提高性能。

voidCPicViewView::OnOdcachehint(NMHDR*pNMHDR,LRESULT*pResult)

{

NMLVCACHEHINT*pCacheHint=(NMLVCACHEHINT*)pNMHDR;

PrepCache(0,min(5,m_e()));

PrepCache(pCacheHint->iFrom,pCacheHint->iTo);

PrepCache(max(0,m_e()-5),m_e());

*pResult=0;

}

在列表需要显示一个项目之前,列表会发送LVN_GETDISPINFO通知,应用程序可以捕获这个消息来提供项目的显示信息。如果显示时需要显示的列表项在缓存中,那么可以从缓存中获取显示信息。否则需要重新从文件获得。

voidCPicViewView::OnGetdispinfo(NMHDR*pNMHDR,LRESULT*pResult)

{

LV_DISPINFO*pDispInfo=(LV_DISPINFO*)pNMHDR;

if(pDispInfo->==-1)return;

HRESULThr=S_OK; LPCITEMIDLISTpidlItem=m_arpFolderItems[pDispInfo->];

CFolderItemInfo*pFolderItemInfo=FindItemInCache(pidlItem);

BOOLbCached=TRUE;

if(pFolderItemInfo==NULL){

bCached=FALSE;

pFolderItemInfo=newCFolderItemInfo;

GetItemInfo(pidlItem,pFolderItemInfo);

}

if(pDispInfo->&LVIF_TEXT){

lstrcpyn(pDispInfo->t,pFolderItemInfo->tszDisplayName,pDispInfo-

Max);

}

if(pDispInfo->&LVIF_IMAGE){

pDispInfo->=pFolderItemInfo->iIcon;

}

if(pDispInfo->&LVIF_STATE){

pDispInfo->=pFolderItemInfo->state;

}

if(!bCached)

deletepFolderItemInfo;

*pResult=0;

}

文件图标的显示

默认情况下,列表项的图标就是其系统图标。首先获得系统图像列表

intCPicViewView::OnCreate(LPCREATESTRUCTlpCreateStruct)

{

>t if(CListView::OnCreate(lpCreateStruct)==-1)

return-1;

HRESULThr=SHGetMalloc(&m_pMalloc);if(FAILED(hr))return-1;

hr=SHGetDesktopFolder(&m_psfDesktop);if(FAILED(hr))return-1;

SHFILEINFOshfi;

ZeroMemory(&shfi,sizeof(SHFILEINFO));

HIMAGELISThi=(HIMAGELIST)SHGetFileInfo(NULL,0,&shfi,sizeof(SHFILEINFO),SHGFI_ICON|SHGFI_SYSICONINDEX|SHGFI_SMALLICON);

GetListCtrl().SetImageList(CImageList::FromHandle(hi),LVSIL_SMALL);

hi=(HIMAGELIST)SHGetFileInfo(NULL,0,&shfi,sizeof

(SHFILEINFO),SHGFI_ICON|SHGFI_SYSICONINDEX|SHGFI_LARGEICON);

GetListCtrl().SetImageList(CImageList::FromHandle(hi),LVSIL_NORMAL);

return0;

}

然后在获取文件信息时,从文件获得其图标在系统图像列表中的索引。

如果列表项是图像文件,并且从文件成功载入图像,那么使用自画功能以替换默认的图标。

voidCPicViewView::OnCustomDraw(NMHDR*pNMHDR,LRESULT*pResult)

{

LPNMLVCUSTOMDRAWlpNMCustomDraw=(LPNMLVCUSTOMDRAW)pNMHDR;

switch(lpNMCustomDraw->Stage){

caseCDDS_PREPAINT:*pResult=CDRF_NOTIFYITEMDRAW;return;

caseCDDS_ITEMPREPAINT:*pResult=CDRF_NOTIFYPOSTPAINT;return;

caseCDDS_ITEMPOSTPAINT:

{

intiItem=lpNMCustomDraw->Spec;

if(iItem==-1){ *pResult=CDRF_DODEFAULT;return;

}

CFolderItemInfo*pItemInfo=FindItemInCache(m_arpFolderItems[iItem]);

if(pItemInfo==NULL||pItemInfo->bFailLoadPic||pItemInfo->pic.m_pPict==NULL){

*pResult=CDRF_DODEFAULT;return;

}

CRectrectIcon;

GetListCtrl().GetItemRect(iItem,&rectIcon,LVIR_ICON);

CDC*pDC=CDC::FromHandle(lpNMCustomDraw->);

pItemInfo->(pDC,rectIcon,rectIcon);

}

*pResult=CDRF_NEWFONT;return;

}

*pResult=0;

}

上面的代码是使用获取的文件显示信息中的图像,在列表项图标的区域画图。

获取显示信息

为了缓存列表项的显示信息,或者显示列表项,需要获取列表项的文字、图标、Overlay图标和缩略图等信息。这里使用了ILCombine来把缓存中的相对PIDL转化为完整的Pidl,再据此获得文件的完整路径,然后调用OleLoadPicturePath函数载入图像。

voidCPicViewView::GetItemInfo(LPCITEMIDLISTpidl,CFolderItemInfo*pItemInfo)

{

HRESULThr=isplayNameOf(pidl,pItemInfo->tszDisplayName);

IShellIcon*pShellIcon=NULL;

hr=m_psfFolder->QueryInterface(IID_IShellIcon,(LPVOID*)&pShellIcon);

if(SUCCEEDED(hr)&&pShellIcon){ pShellIcon->GetIconOf(pidl,0,&pItemInfo->iIcon);

pShellIcon->Release();

}

IShellIconOverlay*pShellIconOverlay=NULL;

hr=m_psfFolder->QueryInterface(IID_IShellIconOverlay,(LPVOID*)

&pShellIconOverlay);

if(SUCCEEDED(hr)&&pShellIconOverlay){

intnOverlay=0;

pShellIconOverlay->GetOverlayIndex(pidl,&nOverlay);

pItemInfo->state=INDEXTOOVERLAYMASK(nOverlay);

pShellIconOverlay->Release();

}

LPITEMIDLISTpidlItemFull=ILCombine(m_pidlFolder,pidl);

if(pidlItemFull){

if(SHGetPathFromIDList(pidlItemFull,pItemInfo->tszPath)){

USES_CONVERSION;

hr=OleLoadPicturePath(

T2OLE(pItemInfo->tszPath)

,NULL,0,RGB(255,255,255)

,IID_IPicture,(LPVOID*)&pItemInfo->pic.m_pPict);

if(FAILED(hr)){

pItemInfo->bFailLoadPic=TRUE;

TRACE("OleLoadPicturePathfailed%s ",pItemInfo->tszPath);

}

}

}

m_pMalloc->Free(pidlItemFull);

}

} 缓存目录的数据

在更改目录时,需要重建目录内容的缓存。这包括目录的pidl和IShellFolder接口指针,目录内容的相对pidl,以及列表项的显示信息(基于性能上的考虑,列表项的显示信息是在接收到LVN_ODCACHEHINT通知的时候缓存的)。

LPITEMIDLISTm_pidlFolder;

IShellFolder*m_psfFolder;

CTypedPtrArraym_arpFolderItems;

CTypedPtrMapm_mapCache;

voidCPicViewView::EnterFolder(LPCITEMIDLISTpidl)

{

USES_CONVERSION;

m_pidlFolder=ILClone(pidl);

if(m_pidlFolder){

LPENUMIDLISTppenum=NULL;

LPITEMIDLISTpidlItems=NULL;

ULONGceltFetched;

HRESULThr;

hr=m_psfDesktop->BindToObject(m_pidlFolder,NULL,IID_IShellFolder,(LPVOID*)

&m_psfFolder);

if(SUCCEEDED(hr)){

hr=m_psfFolder->EnumObjects(NULL,SHCONTF_FOLDERS|SHCONTF_NONFOLDERS,&ppenum);

if(SUCCEEDED(hr)){

while(hr=ppenum->Next(1,&pidlItems,&celtFetched)==S_OK&&(celtFetched)==1){

m_(pidlItems);

}

}

} GetListCtrl().SetItemCount(m_e());

}

}

打开文件夹

本应用程序显示文件夹的内容而不是显示文档的内容,所以我重载了打开文件时的处理,显示目录选择对话框而不是文件打开对话框。

voidCPicViewApp::OnFileOpen()

{

TCHARtszDisplayName[_MAX_PATH];

TCHARtszPathSelected[_MAX_PATH];

LPITEMIDLISTpidlSelected=PidlBrowse(m_pMainWnd->GetSafeHwnd(),0,tszDisplayName);

if(pidlSelected){

if(SHGetPathFromIDList(pidlSelected,tszPathSelected)){

CDocument*pDocument=OpenDocumentFile(tszPathSelected);

pDocument->SetTitle(tszDisplayName);

ILFree(pidlSelected);

}

}

}

注意从外壳调用获得的PIDL一般都需要调用ILFree或者IMalloc::Free释放。一个例外是调用函数SHBindToParent获得的相对pidl,因为它是输入的参数完整pidl的一部分,所以不必另外释放。

在新建或者打开“文件”时候,文档需要通知视图当前文件夹的更改,这是通过调用CDocument::UpdateAllViews和重载CView::OnUpdate实现的。视图对这个通知的处理是清除上一个目录的缓存数据,缓存新目录的数据,以及更新文档标题。

打开文件或者目录

为了使用方便,双击列表项时可以在同一窗口打开子目录,或者调用系统的默认处理程序打开文件。如果文件是快捷方式,那么打开快捷方式的目标。

voidCPicViewView::OnDblclk(NMHDR*pNMHDR,LRESULT*pResult)

{

LPNMLISTVIEWlpnm=(LPNMLISTVIEW)pNMHDR;

if(lpnm->iItem==-1)return;

*pResult=0;

HRESULThr=S_OK;

LPCITEMIDLISTpidlItem=m_arpFolderItems[lpnm->iItem];

LPITEMIDLISTpidlItemFull=ILCombine(m_pidlFolder,pidlItem);

LPITEMIDLISTpidlItemTarget=NULL;

hr=argetFolderIDList(pidlItemFull,&pidlItemTarget);

if(pidlItemTarget){

if(lder(pidlItemTarget)){

CFolderChangeFolderChange;

FolderChange.m_pidlFolder=pidlItemTarget;

OnFolderChange(&FolderChange);

}

else{

SHELLEXECUTEINFOShExecInfo;

=sizeof(SHELLEXECUTEINFO);

=SEE_MASK_IDLIST;

=NULL;

=NULL;

=NULL;

st=pidlItemTarget; meters=NULL;

ctory=NULL;

=SW_MAXIMIZE;

pp=NULL;

ShellExecuteEx(&ShExecInfo);

}

m_pMalloc->Free(pidlItemTarget);

m_pMalloc->Free(pidlItemFull);

}

}

性能的优化

为了更好的用户体验,可以使用自定义的图标大小(这需要完全自行绘制列表项的图标区域),用单独的线程来载入图像,或者使用调整到图标大小的缩略图缓冲(这样每次绘制时不必拉伸图像)。但是这超出了本文的范围。有兴趣的读者可以自己试一下。

参考

需要更多信息的话,可以参考

ShellFAQ

List-ViewControlsOverview

UsingList-ViewControls

CustomizingaControl'sApearanceUsingCustomDraw