2011-05-13 103 views
3

我試圖在Zip存檔的內容可以從Windows資源管理器的Zip文件夾拖放到我的窗口時實現該功能。我實現了IDropTarget的所有必要的方法,並且當我從Windows資源管理器拖放常規文件時,一切正常。將Zip文件夾中的文件拖放到我的窗口

問題發生在下面的方法,當我試圖從一個Zip文件夾中的文件拖動:

HRESULT DragEnter(IDataObject* pDataObject, DWORD grfKeyState, POINTL pt, DWORD* pdwEffect) 
{ 
    static FORMATETC fmtetc_file = {CF_HDROP, 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL}; 
    HRESULT hr = pDataObject->QueryGetData(&fmtetc_file); 
    if(hr == S_OK) 
    { 
     //Format supported 
    } 

    ... 
} 

我得到S_FALSE從的QueryGetData()返回。

有沒有人有任何想法我錯過了什麼?

回答

3

我無法想象Explorer的zip文件處理程序實現了CF_HDROP,因爲它需要它在啓動拖動之前提取文件。我打賭它使用CFSTR_FILEDESCRIPTOR and CFSTR_FILECONTENTS

+0

非常感謝,使用下面的代碼線的dragenter()工作: 'FORMATETC fmtetc_file = {的RegisterClipboardFormat(CFSTR_FILEDESCRIPTOR) ,0,DVASPECT_CONTENT,-1,TYMED_HGLOBAL}; 如果(pDataObject->的QueryGetData(fmtetc_file)== S_OK){ } ' 但現在我有點進退兩難是如何獲得的通道連接到一個zip壓縮文件(其文件被拖進我的窗口)或臨時路徑到文件本身,如果殼牌恰好爲我解壓縮? – ahmd0 2011-05-13 23:58:46

+0

拖放的重點是抽象出zip文件等實現細節;如果你能弄明白的話,我會有點驚訝。 – Luke 2011-05-14 02:33:23

+0

對不起,盧克,但我仍然無法弄清楚。那麼我該如何獲得拖到我窗口中的文件呢? – ahmd0 2011-05-14 08:29:16

1

我想我明白了。你可以審查這個僞代碼,我真的不擅長COM:

Drop(IDataObject* pDataObject, DWORD grfKeyState, POINTL pt, DWORD* pdwEffect) 
{ 
    FORMATETC fmtetc_file_desc = {RegisterClipboardFormat(CFSTR_FILEDESCRIPTOR), 0, DVASPECT_CONTENT, -1, TYMED_HGLOBAL}; 
    if(pDataObject->QueryGetData(&fmtetc_file_desc) == S_OK) 
    { 
     STGMEDIUM stgmed; 
     if(pDataObject->GetData(&fmtetc_file_desc, &stgmed) == S_OK) 
     { 
      if(stgmed.tymed & TYMED_HGLOBAL) 
      { 
       FILEGROUPDESCRIPTOR* pFGD = (FILEGROUPDESCRIPTOR*)::GlobalLock(stgmed.hGlobal); 
       for(int f = 0; f < pFGD->cItems; f++) 
       { 
        STGMEDIUM stgmedFile = {0}; 
        //You may want to move out the RegisterClipboardFormat() API into some global variable 
        FORMATETC fmtetc_file_desc = {RegisterClipboardFormat(CFSTR_FILECONTENTS), 0, DVASPECT_CONTENT, f, TYMED_HGLOBAL | TYMED_ISTREAM | TYMED_ISTORAGE}; 
        if(pDataObject->GetData(&fmtetc_file_desc, &stgmedFile) == S_OK) 
        { 
         BOOL bReadOK = FALSE; 

         if(!bReadOK && (stgmedFile & TYMED_ISTREAM)) 
         { 
          //Now read data from a stream & process it 
          //(If need be, it can be saved in a file) 
          IStream *pstm = pStgmed->pstm; 

          //Size of data in a steam & archived file name 
          STATSTG stg = {0}; 
          SUCCEEDED(pstm->Stat(&stg, STATFLAG_DEFAULT) == S_OK); 

          //Then to read data from a stream 
          //Call repeatedly until all or required data is read) 
          SUCCEEDED(pstm->Read(pStorage, ncbBytesRead, &ucbBytesRead)); 

          //If read and processed successfully 
          bReadOK = TRUE; 


          //Release mem 
          CoTaskMemFree(stg.pwcsName); 

         } 

         //Probably need to implement these as well? 
         if(!bReadOK && (stgmedFile & TYMED_ISTORAGE)) 
         { 
         } 
         if(!bReadOK && (stgmedFile & TYMED_HGLOBAL)) 
         { 
         } 


         ReleaseStgMedium(&stgmedFile);; 
        } 
       } 

       ::GlobalUnlock(stgmed.hGlobal); 
      } 

      ReleaseStgMedium(&stgmed); 

     } 
    } 

} 
+0

對我來說沒關係。在我的測試中,我只見過IStream;不確定是否實際使用了其他存儲方法。此外,Stat()返回E_NOTIMPL,所以你可能只需要Read(),直到cbBytesRead == 0. – Luke 2011-05-14 22:04:06

+0

嗯,真的。你在測試什麼版本的Windows? – ahmd0 2011-05-14 22:06:53

+0

我正在測試XP SP3 – Luke 2011-05-15 00:59:22

相關問題