青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

一個定制CFileDialog對話框的實例

很多程序員都喜歡讓自己的代碼運行效果與眾不同。Windows系統(tǒng)的應(yīng)用程序打開某個文件一般使用的都是默認的CFileDialog。但是這個默認的CFileDialog往往滿足不了用戶的要求。我就碰到一個這樣的用戶,他的要求如下:
  • 1、在默認的CFileDialog對話框中加一個預(yù)覽窗格,以便在選中ASCII文件時能看到所選文件的內(nèi)容,也就是用*.txt作為文件過濾條件。
  • 2、在默認的CFileDialog對話框中加一個"全部"按鈕來選擇某個目錄中所有的.txt文件。
  • 3、如果選擇的目錄中沒有.txt文件時,要將"全部"按鈕disable,也就是置灰這個按鈕。
   實現(xiàn)上面這些需求必須要改裝CFileDialog對話框。當(dāng)最后寫完程序時,功能到是全都實現(xiàn)了,但在Windows 2000環(huán)境測試中,用戶發(fā)現(xiàn)了這樣一個問題:如果先選中某個文件,然后再去選某個文件夾,預(yù)覽窗格仍然顯示的那個文件的內(nèi)容。盡管在OnFileNameChange中對CDN_SELCHANGE進行了處理,為了獲取所選的文件/路徑名,也調(diào)用了CFileDialog::GetPathName。但是即使是選中了文件夾,GetPathName仍然返回的是文件的名字。即便嘗試用其它的通知消息和函數(shù),比如 CDN_FOLDERCHANGE 和 GetFileName,但仍舊存在同樣的問題。必須承認在Windows 2000中,CFileDialog是個不完美的對話框,確實存在上述問題。正是有這些不完美,程序員們才忙得個不亦樂乎......那么到底如何判斷用戶選中的是文件還是文件夾呢?下面就讓我們從用戶需求開始,一個一個解決所碰到的問題。
    首先簡單介紹下本文引入的三個輔助類:CFileDialogHook;CFileDialogOwnerHook和CFileDlgHelper,這三個類很簡單,其功能分別是:子類化文件對話框;子類化文件對話框的父窗口或宿主窗口,這兩個類只在CFileDlgHelper類中使用,一些重要的處理都在CFileDlgHelper中。它的使用方法很簡單,實例化CFileDlgHelper以后調(diào)用Init即可。
class CMyOpenDlg ... {
protected:
CFileDlgHelper m_dlghelper;//實例化
};
BOOL CMyOpenDlg::OnInitDialog()
{
m_dlghelper.Init(this)//初始化
……
}     
    初始化CFileDlgHelper以后,便可以用它來獲取列表控制以及判斷選項是否有文件夾屬性,例如:
CListCtrl* plc = m_dlghelper.GetListCtrl();
POSITION pos = plc->GetFirstSelectedItemPosition();
while (pos) {
int i = plc->GetNextSelectedItem(pos);
if (fdh.IsItemFolder(i)) {
// 顯示"(FOLDER)"……
} else {
// 顯示其它內(nèi)容
}
}

    毫無疑問,要改裝CFileDialog對話框,必須建立一個它的派生類以及一個新的對話框資源。“全部”按鈕的實現(xiàn)代碼是這樣的:

void CMyOpenDlg::OnSelectAll()
{
CListCtrl* plc = m_dlghelper.GetListCtrl();
for (int i=0; i<plc->GetItemCount(); i++) {
CString fn = plc->GetItemText(i,0);
if (IsTextFileName(fn)) {
plc->SetItemState(i,LVIS_SELECTED,
LVIS_SELECTED);
}
}
plc->SetFocus();
}
    當(dāng)所選目錄中沒有.txt文件時,要disable“全部”按鈕的處理稍微麻煩一些,要用到ON_UPDATE_COMMAND_UI消息。回顧一下MFC有關(guān)UI更新的基本方法,通常是在主消息循環(huán)處于空閑狀態(tài)時候——也就是說在消息隊列中沒有待處理的消息。但對話框則有所不同,尤其是運行模式對話框時,MFC啟動另外一個消息循環(huán)。當(dāng)沒有消息等待處理的時候,CWnd::DoModal向?qū)υ捒虬l(fā)送一個WM_KICKIDLE消息。所以要想讓對話框處理UI,常用的方式是這樣的:
LRESULT CMyDialog::OnKickIdle(WPARAM wp, LPARAM lp)
{
UpdateDialogControls(this, TRUE);
return 0;
}      
    CWnd::UpdateDialogControls將神奇的CN_UPDATE_COMMAND_UI消息發(fā)送到對話框,觸發(fā)ON_UPDATE_COMMAND_UI處理例程。可惜這個方法對CFileDialog對話框不靈。原因是CFileDialog重寫了DoModal,它不會以正常方式運行某個消息循環(huán),而是調(diào)用::GetOpenFileName (或::GetSaveFileName)。這些API函數(shù)都有自己消息循環(huán),并且你無法鉆進去進行消息空閑處理。無論什么時候,每當(dāng)模式對話框處于等待消息狀態(tài)時,對話框發(fā)送自己的WM_ENTERIDLE消息。從這里進去才可以處理UI更新事宜。但有幾個細節(jié)需要注意。首先,Windows只發(fā)送WM_ENTERIDLE消息到對話框的所有者——此處為主框架——所以必須在那里捕獲這個消息。然后,只要對話框仍然處于空閑狀態(tài),則Windows繼續(xù)發(fā)送WM_ENTERIDLE,但只需要調(diào)用UpdateDialogControls一次,此間可以進行常規(guī)的標(biāo)志設(shè)置。那到底什么時候設(shè)置標(biāo)志呢?無論何時,UI狀態(tài)的改變,都是在對話框獲得到WM_COMMAND 或 WM_NOTIFY消息之后。所以還必須在CFileDialog派生的對話框中截獲這些消息。 因為這些都是一些繁瑣的細節(jié),所以最好將它們封裝到在一個新類中,這就是CFileDlgHelper的來由。只要從CFileDialog派生的對話框OnInitDialog函數(shù)中調(diào)用CFileDlgHelper的Init,便不用操心ON_UPDATE_COMMAND_UI的處理細節(jié)。CFileDlgHelper是如何實現(xiàn)的呢?告訴你吧,利用萬能類CSubclassWnd,這個類可以用Windows的方式子類化任何窗口,通過在某個窗口過程之前安裝一個新的窗口過程來實現(xiàn)消息的捕獲。實際上,CFileDlgHelper 用了兩個CSubclassWnds派生類:一個用來截獲發(fā)送到對話框父窗口的WM_ENTERIDLE消息,另一個用來截獲發(fā)送到對話框本身的WM_COMMAND 或 WM_NOTIFY。當(dāng)主窗口得到WM_ENTERIDLE消息時,CFileDialogOwnerHook解釋它并更新對話框控制:
LRESULT CFileDialogOwnerHook::WindowProc(...)
{
if (msg==WM_ENTERIDLE) {
if (m_pHelper->m_bUpdateUI) {
m_pDlg->UpdateDialogControls(m_pDlg, FALSE);
m_pHelper->m_bUpdateUI=FALSE;
}
}
return CSubclassWnd::WindowProc(msg, wp, lp);
}
當(dāng)對話框得到WM_NOTIFY 或者WM_COMMAND消息時,CFileDialogHook重置標(biāo)志。
LRESULT CFileDialogHook::WindowProc(UINT msg, WPARAM wp, LPARAM lp)
{
if (msg==WM_COMMAND || msg==WM_NOTIFY) {
m_pHelper->m_bUpdateUI = TRUE;
}
return CSubclassWnd::WindowProc(msg, wp, lp);
}     
一旦知道了其中的奧秘,一切就這么簡單。注意從CSubclassWnd派生了兩個類——CFileDialogOwnerHook和CFileDialogHook,一個用來對付主框架,另一個用來對付對話框本身,它們都在隱含在CFileDlgHelper類中。有了它,“按鈕”的UI更新就會象你所期望的那樣:
void CMyOpenDlg::OnUpdateSelectAll(CCmdUI* pCmdUI)
{
CFileDlgHelper& fdh = m_dlghelper;
CListCtrl* plc = fdh.GetListCtrl();
for (int i=0; i<plc->GetItemCount(); i++) {
if (IsTextFileName(fdh.GetItemName(i))) {
pCmdUI->Enable(TRUE);
return;
}
}
pCmdUI->Enable(FALSE);
}      
   以上是用戶需求的實現(xiàn),下面來解決Window 2000環(huán)境測試出現(xiàn)的問題:如何確定在列表框中選擇的是文件還是文件夾。
    要想解決這個問題,就必須關(guān)注對話框中的列表控制(ListCtrl/ListView),許多普通的對話框里的控制都有明確的IDs,如靜態(tài)文本控制有stc1,以及列表框有l(wèi)st1,這些符號都定義在文件中。你可以把列表控制看成是lst1,但用Spy++察看后,如圖一所示:


圖一 Spy++

你會發(fā)現(xiàn)列表控制實際上被包含在另一個窗口類SHELLDLL_DefView中。SHELLDLL_DefView窗口的ID為lst2,其項下的列表控制(SysListView32)的子ID為1。所以,為了要得到這個列表控制,可以這樣編碼:

// 在自己的CFileDialog 派生類中
CListCtrl* plc = (CListCtrl*)GetParent()->GetDlgItem(lst2)->GetDlgItem(1);      
    記住,在定制CFileDialog時,它實際上是一個實際對話框的子對話框,這就是必須用GetParent的原因。更多的細節(jié)請參考MSDN中的相關(guān)文章。強制類型轉(zhuǎn)換 CListCtrl* 與每一個常見的MFC訣竅一樣,因為CListCtrl既沒有數(shù)據(jù)成員也沒有虛擬函數(shù)成員,它是一個純粹的包裝類(因為GetDlgItem返回一個臨時的CWnd指針,而不是CListCtrl,每次碰到這種情況,常常都會讓人感到沮喪,其實這很正常)。 一旦你有了列表控制的指針,便可以做任何想做事情——例如獲取選中的路徑名,調(diào)用CListCtrl::GetItemText并添加結(jié)果到當(dāng)前打開的文件夾(GetFolderPath/CDM_GETFOLDERPATH)。有了路徑名,如何知道它到底時文件還是文件夾呢?方法如下:
#include 
// 檢查路徑名是不是文件夾
static BOOL IsFolder(LPCTSTR pathname)
{
struct stat st;
return stat(pathname, &st)==0 && (st.st_mode & _S_IFDIR);
}
    這里需要注意的是:不管怎樣,如果路徑名不是文件夾,你也不能因此就斷定它就是一個文件!因為它還可能是其它的外殼對象,如"網(wǎng)上鄰居"或者"我的電腦"之類的東西。 詳細做法可以參考本文的例子程序 OpenFileDlg,它還示范了如何建立預(yù)覽對話框。這個程序可以進行多項選擇,如果只選中一個.txt文件,則預(yù)覽窗格顯示文件的開始幾行。程序還帶一個調(diào)試窗口,窗口中列出選中的條目,如果選中的是文件夾,則在它的旁邊會有“FOLDER”說明。如圖二所示。


圖二運行中的OpenFileDlg

    如果選中的是文件夾,則OpenFileDlg會清空預(yù)覽格,這樣就解決了本文所提出的預(yù)覽問題。當(dāng)然,如果運行環(huán)境是Windows XP,而非Windows 2000,那么就不會碰上這個問題!在Windows XP中,OnFileNameChange/CDN_SELCHANGE會返回正確的文件名和文件夾名字。但仍然可以用CFileDlgHelper類獲取列表控制,選項名稱等。并且仍然需要IsFolder來檢查路徑名是不是文件夾。
    其實,在OnSelectAll處理代碼中,IsTextFileName的功能是查找以.txt結(jié)尾文件名字。這個函數(shù)真的能實現(xiàn)這個功能嗎?其實,在程序中有個致命的問題——如果用戶定制了資源管理器來隱藏已知文件類型的擴展名。那么,.txt就不會出現(xiàn)在列表框中。也就是說CFileDlgHelper::GetItemName返回foo,而不是foo.txt。實際上,如果擴展名被隱藏,那么象foo.txt、foo.jpg和foo.doc等等這樣的文件都以名字foo出現(xiàn)(試一下就知道了)。如此一來,怎么知道這個foo文件到底是此foo,還是彼foo呢?問題真是解決不完啊,搞掂這個問題,又出那個問題。唉,好累啊,下次再說吧......

posted on 2008-04-26 08:10 wrh 閱讀(931) 評論(0)  編輯 收藏 引用


只有注冊用戶登錄后才能發(fā)表評論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


導(dǎo)航

<2008年4月>
303112345
6789101112
13141516171819
20212223242526
27282930123
45678910

統(tǒng)計

常用鏈接

留言簿(19)

隨筆檔案

文章檔案

收藏夾

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国产精品一区在线观看| 香蕉尹人综合在线观看| 亚洲午夜一区二区| 一区二区三区日韩在线观看| 亚洲精品看片| 亚洲一区二区免费看| 欧美一级视频一区二区| 午夜精品一区二区三区电影天堂 | 亚洲另类一区二区| 99热这里只有精品8| 99国产精品久久久久久久久久 | 欧美体内she精视频| 国产精品视频网| 一区在线视频| 在线一区二区三区做爰视频网站| 亚洲天天影视| 麻豆精品一区二区综合av| 91久久国产自产拍夜夜嗨| 亚洲国产婷婷综合在线精品| 亚洲高清久久网| 亚洲伊人一本大道中文字幕| 久久蜜臀精品av| 亚洲理论电影网| 久久久www成人免费毛片麻豆| 免播放器亚洲一区| 国产精品美女| 亚洲国产精品久久人人爱蜜臀| 在线视频亚洲| 你懂的国产精品| 亚洲一区二区成人| 欧美岛国在线观看| 国产亚洲精久久久久久| 这里只有视频精品| 免费观看亚洲视频大全| 亚洲综合好骚| 欧美日韩一本到| 亚洲精品视频免费| 欧美a级在线| 欧美影院在线播放| 欧美偷拍一区二区| 亚洲精品中文字幕有码专区| 久久综合亚州| 欧美在线视频日韩| 国产精品亚洲成人| 亚洲一区二区三区久久| 亚洲日韩成人| 欧美a级片一区| 永久免费视频成人| 欧美在线亚洲| 亚洲欧美第一页| 国产精品人人做人人爽| 亚洲一区二区少妇| 日韩天堂av| 欧美日韩国产一区二区三区| 91久久夜色精品国产九色| 久久在线免费观看视频| 欧美一区观看| 国产综合久久| 久久综合99re88久久爱| 欧美一区二区三区在线视频| 国产欧美日韩综合一区在线观看 | 国语自产在线不卡| 久久精品国产亚洲精品| 亚洲女同性videos| 国产精品揄拍一区二区| 欧美一二三区在线观看| 亚洲免费在线观看| 国产亚洲aⅴaaaaaa毛片| 久久精品国产69国产精品亚洲| 亚洲欧美日韩另类| 国产一区二区高清不卡| 久久天堂精品| 美女精品网站| 一区二区三区精品久久久| 99成人精品| 国产精品一区二区在线观看不卡| 国内外成人在线| 久久久亚洲午夜电影| 狂野欧美激情性xxxx欧美| 亚洲精品国产精品国自产在线 | 亚洲国产午夜| 亚洲免费观看| 国产丝袜美腿一区二区三区| 久久综合给合| 欧美精品一区在线观看| 亚洲欧美日韩国产一区二区| 欧美一级一区| 日韩网站免费观看| 亚洲欧美日韩成人| 亚洲高清视频在线观看| 99这里有精品| 激情久久中文字幕| 亚洲人成高清| 国产一区视频在线观看免费| 欧美aa在线视频| 国产精品久久二区二区| 蜜桃av一区二区三区| 欧美色中文字幕| 老色鬼精品视频在线观看播放| 欧美精品18videos性欧美| 欧美在线播放视频| 欧美国产日韩一区| 久久精品国产视频| 欧美日韩精品免费观看视频| 久久九九全国免费精品观看| 欧美好骚综合网| 久久野战av| 国产欧美在线视频| 日韩视频在线观看| 亚洲片区在线| 欧美在线一级视频| 亚洲天堂久久| 欧美成年人网站| 久久久久久久综合色一本| 欧美视频日韩| 亚洲国产日本| 亚洲福利专区| 久久久www成人免费毛片麻豆| 亚洲一区二区在线播放| 欧美成年人在线观看| 久久久噜噜噜久噜久久| 国产精品美女久久久久久免费 | 久久激情视频| 亚洲欧美日韩国产| 欧美精品网站| 亚洲二区免费| 亚洲国产小视频| 久久视频在线视频| 久久网站免费| 韩国成人福利片在线播放| 亚洲自拍偷拍视频| 亚洲欧美欧美一区二区三区| 欧美精品综合| 亚洲精品男同| 一区二区免费在线观看| 欧美精品色网| 日韩视频免费观看高清完整版| 亚洲日本无吗高清不卡| 蜜臀a∨国产成人精品| 美女黄毛**国产精品啪啪 | 一区二区三区欧美视频| 一本一本久久a久久精品综合麻豆 一本一本久久a久久精品牛牛影视 | 久热爱精品视频线路一| 国产午夜精品一区二区三区视频| 亚洲天堂男人| 欧美在线视频一区| 国产婷婷色一区二区三区| 久久se精品一区二区| 免费久久99精品国产自| 91久久久久久国产精品| 欧美猛交免费看| 亚洲一区在线播放| 久久天堂av综合合色| 亚洲青涩在线| 国产精品分类| 久久精品视频免费播放| 亚洲国产老妈| 亚洲欧美久久| 在线观看视频欧美| 欧美屁股在线| 欧美一区观看| 亚洲日本成人| 欧美在线观看一二区| 精久久久久久| 欧美日韩一区二区三区四区在线观看| 一区二区日韩免费看| 久久久女女女女999久久| 亚洲精品在线视频| 国产美女高潮久久白浆| 久久人人看视频| 一本色道久久88综合亚洲精品ⅰ| 久久九九热re6这里有精品| 亚洲美女中文字幕| 国产日韩欧美三区| 欧美激情综合五月色丁香小说 | 午夜伦欧美伦电影理论片| 欧美成人黑人xx视频免费观看| 99精品欧美一区二区三区| 国产欧美精品日韩精品| 欧美大片91| 午夜精品亚洲| 99国产精品视频免费观看一公开 | 国产一区二区激情| 欧美国产日韩在线| 久久国产精品99国产精| 日韩亚洲一区在线播放| 另类av一区二区| 午夜精品短视频| 日韩亚洲一区在线播放| 激情六月综合| 国产日产欧产精品推荐色 | 亚洲永久精品国产| 亚洲精品久久在线| 在线成人h网| 国产三级精品在线不卡| 欧美日韩亚洲三区| 欧美精品v日韩精品v韩国精品v | 亚洲一区二区伦理| 亚洲一二三区在线| 亚洲精品视频在线播放|