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

2016-02-19 14:26 230 1 收藏

今天图老师小编给大家精心推荐个VC一点通:实现文件夹的缩略图显示教程,一起来看看过程究竟如何进行吧!喜欢还请点个赞哦~

【 tulaoshi.com - 编程语言 】

   本示例演示了列表控件的虚列表和自画功能,也演示了一些系统外壳的函数和接口的使用方法。
  
  单击这里下载本文的代码。
  
  预备性阅读

  在阅读本文之前,建议先对列表视图控件和系统外壳有一个基本的了解。建议阅读以下SDK文章
  
  ShellFAQ
  List-ViewControlsOverview
  UsingList-ViewControls
  CustomizingaControl'sAppearanceUsingCustomDraw

  创建应用程序

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

  本文采用虚列表技术,使得显示信息是在第一次显示的时候才被获取。为了创建虚列表,在创建之前需要指定列表的风格
  
  BOOLCPicViewView::PreCreateWindow(CREATESTRUCT&cs)
  {
  cs.style&=~LVS_TYPEMASK;
  cs.style|=LVS_ICON|LVS_OWNERDATA;
  returnCListView::PreCreateWindow(cs);
  }
  同时,因为列表项的Overlay图标也是被动态获取的,所以需要设置动态Overlay图标
  
  voidCPicViewView::OnInitialUpdate()
  {
  CListView::OnInitialUpdate();
  GetListCtrl().SetCallbackMask(LVIS_OVERLAYMASK);
  }
  
  缓存显示信息

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)

  在列表需要显示一个范围的项目之前,列表会发送LVN_ODCACHEHINT通知,应用程序可以捕获这个消息来缓存部分列表的显示信息,以提高性能。
    
  voidCPicViewView::OnOdcachehint(NMHDR*pNMHDR,LRESULT*pResult)
  {
  NMLVCACHEHINT*pCacheHint=(NMLVCACHEHINT*)pNMHDR;
  PrepCache(0,min(5,m_arpFolderItems.GetSize()));
  PrepCache(pCacheHint-iFrom,pCacheHint-iTo);
  PrepCache(max(0,m_arpFolderItems.GetSize()-5),m_arpFolderItems.GetSize());
  *pResult=0;
  }
 在列表需要显示一个项目之前,列表会发送LVN_GETDISPINFO通知,应用程序可以捕获这个消息来提供项目的显示信息。如果显示时需要显示的列表项在缓存中,那么可以从缓存中获取显示信息。否则需要重新从文件获得。
  
  voidCPicViewView::OnGetdispinfo(NMHDR*pNMHDR,LRESULT*pResult)
  {
  LV_DISPINFO*pDispInfo=(LV_DISPINFO*)pNMHDR;
  if(pDispInfo-item.iItem==-1)return;
  HRESULThr=S_OK;
  LPCITEMIDLISTpidlItem=m_arpFolderItems[pDispInfo-item.iItem];
  CFolderItemInfo*pFolderItemInfo=FindItemInCache(pidlItem);
  BOOLbCached=TRUE;
  if(pFolderItemInfo==NULL){
  bCached=FALSE;
  pFolderItemInfo=newCFolderItemInfo;
  GetItemInfo(pidlItem,pFolderItemInfo);
  }
  if(pDispInfo-item.mask&LVIF_TEXT){
  lstrcpyn(pDispInfo-item.pszText,pFolderItemInfo-tszDisplayName,pDispInfo-  item.cchTextMax);
  }
  if(pDispInfo-item.mask&LVIF_IMAGE){
  pDispInfo-item.iImage=pFolderItemInfo-iIcon;
  }
if(pDispInfo-item.mask&LVIF_STATE){
pDispInfo-item.state=pFolderItemInfo-state;
  }
  if(!bCached)
  deletepFolderItemInfo;
  *pResult=0;
  }
  

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)文件图标的显示

  默认情况下,列表项的图标就是其系统图标。首先获得系统图像列表
  
  intCPicViewView::OnCreate(LPCREATESTRUCTlpCreateStruct)
  {
  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-nmcd.dwDrawStage){
  caseCDDS_PREPAINT:*pResult=CDRF_NOTIFYITEMDRAW;return;
  caseCDDS_ITEMPREPAINT:*pResult=CDRF_NOTIFYPOSTPAINT;return;
  caseCDDS_ITEMPOSTPAINT:
  {
  intiItem=lpNMCustomDraw-nmcd.dwItemSpec;
  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-nmcd.hdc);
  pItemInfo-pic.Render(pDC,rectIcon,rectIcon);
  }
  *pResult=CDRF_NEWFONT;return;
  }
  *pResult=0;
  }
  
  上面的代码是使用获取的文件显示信息中的图像,在列表项图标的区域画图。

获取显示信息

  为了缓存列表项的显示信息,或者显示列表项,需要获取列表项的文字、图标、Overlay图标和缩略图等信息。这里使用了ILCombine来把缓存中的相对PIDL转化为完整的Pidl,再据此获得文件的完整路径,然后调用OleLoadPicturePath函数载入图像。
  
 voidCPicViewView::GetItemInfo(LPCITEMIDLISTpidl,CFolderItemInfo*pItemInfo)
  {
  HRESULThr=theApp.SHGetDisplayNameOf(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_arpFolderItems.Add(pidlItems);
  }
  }
  }
  GetListCtrl().SetItemCount(m_arpFolderItems.GetSize());
  }
  }
  

打开文件夹

  本应用程序显示文件夹的内容而不是显示文档的内容,所以我重载了打开文件时的处理,显示目录选择对话框而不是文件打开对话框。
  
  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=theApp.SHGetTargetFolderIDList(pidlItemFull,&pidlItemTarget);
  if(pidlItemTarget){
  if(theApp.ILIsFolder(pidlItemTarget)){
  CFolderChangeFolderChange;
  FolderChange.m_pidlFolder=pidlItemTarget;
  OnFolderChange(&FolderChange);
  }
  else{
  SHELLEXECUTEINFOShExecInfo;
  ShExecInfo.cbSize=sizeof(SHELLEXECUTEINFO);
  ShExecInfo.fMask=SEE_MASK_IDLIST;
  ShExecInfo.hwnd=NULL;
  ShExecInfo.lpVerb=NULL;
  ShExecInfo.lpFile=NULL;
  ShExecInfo.lpIDList=pidlItemTarget;
  ShExecInfo.lpParameters=NULL;
  ShExecInfo.lpDirectory=NULL;
  ShExecInfo.nShow=SW_MAXIMIZE;
  ShExecInfo.hInstApp=NULL;
  ShellExecuteEx(&ShExecInfo);
  }
  m_pMalloc-Free(pidlItemTarget);
  m_pMalloc-Free(pidlItemFull);
  }
  }  
  
  性能的优化
  为了更好的用户体验,可以使用自定义的图标大小(这需要完全自行绘制列表项的图标区域),用单独的线程来载入图像,或者使用调整到图标大小的缩略图缓冲(这样每次绘制时不必拉伸图像)。但是这超出了本文的范围。有兴趣的读者可以自己试一下。
  
  参考
  需要更多信息的话,可以参考
  
  ShellFAQ
  List-ViewControlsOverview
  UsingList-ViewControls
  CustomizingaControl'sApearanceUsingCustomDraw

来源:http://www.tulaoshi.com/n/20160219/1606960.html

延伸阅读
标签: 电脑入门
一、如何使用word缩略图 要想使用word的缩略图,仅需要一步简单的操作即可,如下图一所示。 点击word界面中的菜单视图→缩略图,这样就能使用该功能了。 一旦你使用了该功能,那么,每次打开word都会处于缩略图状态之下,有时候并不需要缩略图功能,因此,可通过再次点击视图→缩略图菜单取消该功能的使用。 二、缩略图的功能 ...
VC++实现对文件夹时间的修改 摘要: 本文给出了一种通过VC++程序对文件夹的时间属性信息进行获取与更改的一种方法。 关键词: 文件夹;时间属性;Visual C++ 6.0 引言: 通常在数据维护与数据管理等工作中,经常要对各类数据进行备份操作。对于大多数的IT企业和绝大多数的个人用户而言,数据的备份主要是对数据内容...
powerpoint自动缩略图效果详解   您相信实现多张图片都能在一个幻灯片上演示吗,一个幻灯片演示文稿吗? 单击自动缩放效果,单击恢复再一次。方法是: 新演示文稿,单击"插入"菜单中的"对象"命令,选择"Microsoft PowerPoint 演示文稿",在插入的演示演示文稿对象中插入网站的图片,将图片的...
标签: windows 操作系统
在Windows XP/2003中,系统为我们提供了一种显示文件的更直观的方式——缩略图,在这种显示方式下,图片、文本类型文件等的缩略内容会替代文件的图标,这样不用打开文件就能看到它的大致内容,非常方便。但是这种方式也会占用大量的系统资源。如果你的计算机性能不是非常强大,可以通过修改注册表的方法来调整缩略图的大小和质量,从而调整...
标签: windows系统
Win8缩略图显示错乱如何恢复   Win8缩略图显示错乱恢复的方法如下: 在Windows 8系统中浏览图片或视频文件时。有时图标,缩略图会无法正常显示,通常这样的情况若不是中毒了,就是相关缓存文件出了问题。 浏览文件时显示的缩略图,一般都存储在缓存文件夹中,以方便下次更快速地浏览,但有时缓存文件会崩溃,这时就需要...

经验教程

877

收藏

35
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部