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

渴望飛翔
Fly in the C++ Sky...
posts - 9,  comments - 6,  trackbacks - 0

做一個MFC程序的時候碰到一個需求。就是需要根據定制情況,動態生成菜單,菜單的具體結構和信息是之前不知道的(因此不能利用工具構造),點擊不同類型的菜單會觸發特定的一類事件(需要動態綁定事件)。這種需求實際是蠻不BT的,很多場合下都可能會有,用C#寫了個Demo花了不到半個小時。但轉到MFC下來寫,就費盡周折。其實這個問題代表了在MFC中動態創建資源綁定事件的一般性問題,所以總結一下。

動態創建菜單需要先了CMenu類。通常我們利用工具繪制一個菜單,每一個菜單項下都可以視為有一個CMenu類。它們聯系在一起,形成樹狀。典型的一個菜單對應過來是如下圖這個樣子:

 

如上,CMenu可以分成三種,一個是Popup(黃色),一個是Separator(灰色),一個是Item(紅色)。前兩種都是沒有ID信息的,Popup有一個指針,指向其SubMenu;Item保存各種信息有ID可以響應事件;Separator,恩,基本是一窮二白的。

CMenu的CreateMenu方法可以創建一個菜單資源,用DeleteMenu(包含所有子菜單)或DestoryMenu可以銷毀菜單資源,用AppendMenu可以添加一個菜單。了解這些內容,就可以開工了,現實現上圖所示的MainSubMenu1下菜單的動態創建,代碼如下:

    // 假設在ChildFrm中,調用該方法獲得當前的主菜單指針
    CMenu* mainMenu = AfxGetMainWnd()->GetMenu();
    CMenu* subMenu = NULL;


    // 遍歷主菜單下的各級菜單尋找名為MainSubMenu1的菜單

    int menuCount = mainMenu->GetMenuItemCount();

    for(int i = 0; i < menuCount; i++)
    {
        CString menuName;
        if(mainMenu->GetMenuStringA(i, menuName, MF_BYPOSITION)
            && menuName == "&MainSubMenu1")
        {
            drawingMenu = mainMenu->GetSubMenu(i);
            break;
        }
    }

    // 移除原有的菜單項
    int subMenu1Count = subMenu->GetMenuItemCount();
    for(int i = subMenu1Count - 1; i >= 0; i--)
    {
        subMenu->DeleteMenu(i, MF_BYPOSITION);
    }

    // 動態添加Item菜單項
    for(int i = 0; i < 2; i++)
    {

        CString message = "";

        subMenu->AppendMenuA(MF_STRING, ID_BEGIN + i, message.Format("SubSubMenu%i", i);

    }


    // 添加分隔符

    subMenu->AppendMenuA(MF_SEPARATOR);


    // 添加彈出式子菜單

    CMenu * popupMenu = new CMenu();
    popupMenu->CreateMenu();
    for(int i = 0; i < 2; i++)
    {

        CString message = "";

        popupMenu->AppendMenuA(MF_STRING, ID_BEGIN + 2 + i, message.Format("PopupSubMenu%i", i));

    }
    subMenu->AppendMenuA(MF_POPUP, (UINT_PTR)popupMenu->operator HMENU(), "PopupMenu");

有幾個需要注意的地方,一個是主菜單的指針獲得,可以參考《MFC框架各部分指針獲取方式》一文。另一個是Popup的菜單建立,策略是分成兩部分,先new出內存在Create出資源,缺一不可。最后一個是為每個Item菜單合理分配ID,這些ID須事先預留出來,在MFC中,至少40000到49000通常都是沒人用。

這也就引出下一個問題,即菜單事件的動態綁定。我們知道在.net中,事件是真正動態綁定的,而MFC中的事件都是只能靜態綁定,這是由兩者的編譯方式決定的。所以,在MFC中需要定義菜單事件,你需要先挖好坑(預留足夠ID),規定每個坑種什么羅卜(將不同類型的ID綁定到不同類別的事件處理函數上),最后才能按坑種羅卜(為執行相應事件的菜單設置相應的ID)。

可以有兩種方式來綁定對應ID處理的事件,一個是通過ON_COMMAND_RANGE宏(想一下ON_COMMAND宏會不會派上用場?)在MessageMap里綁定批量處理事件的函數;另一個是重載PreTranslateMessage函數,截獲并判斷ID來進行處理。思想都是類似的。值得注意的是,通常還需要配套使用ON_UPDATE_COMMAND_UI_RANGE來保證動態創建的菜單Enable為True,否則很可能菜單不可以點擊(我就被郁悶過很久:()。



duguguiyu 2007-07-21 19:34 發表評論

文章來源:http://www.cnblogs.com/duguguiyu/archive/2007/07/21/826816.html
posted @ 2007-07-21 19:34 duguguiyu 閱讀(3391) | 評論 (4)編輯 收藏
前人在CSDN總結的,曾經幫助過我,整理總結一下,希望也能幫助一下別人。
    

 

獲得CWinApp

獲得CMainFrame

獲得CChildFrame

獲得CDocument

獲得CView

在CWinApp中

 

AfxGetMainWnd()

m_pMainWnd

AfxGetMainWnd()->MDIGetActive()

AfxGetMainWnd()->GetActiveFrame()

SDI:AfxGetMainWnd()->GetActiveView()->GetDocument()

MDI:AfxGetMainWnd()->MDIGetActive()->GetActiveView()->GetDocument()

SDI:AfxGetMainWnd()->GetActiveView()  
MDI:AfxGetMainWnd()->MDIGetActive()->GetActiveView() 
在CMainFrame中

AfxGetApp()

theApp

MDIGetActive()

GetActiveFrame()

SDI:GetActiveView()->GetDocument()  
MDI:MDIGetActive()->GetActiveView()->GetDocument()  
SDI:GetActiveView()  
MDI:MDIGetActive()->GetActiveView() 
在CChildFrame中

AfxGetApp()

theApp

GetParentFrame() 

 

GetActiveView()->GetDocument()   GetActiveView()
在CDocument中

AfxGetApp()

theApp

AfxGetMainWnd()  

AfxGetMainWnd()->MDIGetActive()

AfxGetMainWnd()->GetActiveFrame()

POSITION   pos   =   GetFirstViewPosition();GetNextView(pos)  
在CView中

AfxGetApp()

theApp

AfxGetMainWnd()   GetParentFrame()   GetDocument()
在其他類中

AfxGetApp()

AfxGetMainWnd()  

AfxGetMainWnd()->MDIGetActive()

AfxGetMainWnd()->GetActiveFrame() 

SDI:AfxGetMainWnd()->GetActiveView()->GetDocument()

MDI:AfxGetMainWnd()->MDIGetActive()->GetActiveView()->GetDocument()

SDI:AfxGetMainWnd()->GetActiveView()  
MDI:AfxGetMainWnd()->MDIGetActive()->GetActiveView() 
 
理一理MFC的這幾個類的關系,可以很容易明白上面的這些亂七八糟的邏輯。
App是應用域,所有的域中的東西都可以通過全局函數訪問到它。
MainFrame是主框架,也基本可以用全局函數訪問到。
MainFrame下是若干個ChildFrame,ChildFrame中若干個View和Document(可能不成對),ChildFrame管理著View,View和Document進行互操作。
因此整體框架就出來了,一般除了直接應用的關系都可以通過MainFrame-->Active ChildFrame-->Active View-->Document這條線進行訪問,這應該叫什么來自?萬能方法吧^_^。
恕我懶惰,不愿意畫一個更詳細的圖解,湊合著看看吧。



duguguiyu 2007-06-22 01:33 發表評論

文章來源:http://www.cnblogs.com/duguguiyu/archive/2007/06/22/792511.html
posted @ 2007-06-22 01:33 duguguiyu 閱讀(498) | 評論 (0)編輯 收藏

最近在寫一些關于MFC的東西,是因為做了些MFC的項目,一些零散的東西需要總結一下。但這并不代表我有些喜歡他了,其實還是蠻討厭的。畢竟看了.Net Framework,再看N多年前的MFC,多少是有些不順眼的。機理上的東西不敢多說,有些變量和方法名字的設置,站在FCL的設計角度看了,多少覺得有些不爽。也許是我太弱,也許是因為它的設計上考慮還是沒有足夠的經驗,列舉一些,娛樂娛樂。

1. 在構造一個自定義的Dialog對象時,我們會傳入一個CWnd的指針進去,那個名為pParent的對象。但當我在Dialog中調用GetParent函數的時候,無論傳進的是什么指針(比如CYourView),得到的返回值都是指向CMainFrame的指針。曾經由于這個我調了N久的程序,我承認這是由于我對MFC的了解不夠造成的,但這名字也太具有迷惑性了吧。

2. 當然還有臭名昭著的UpdateData函數,很榮幸在這點上我和大師一樣,每次用到的時候都要停下來查看一下才能分清楚那個詭異的bool變量的意義。也許這在有IDE的時候不是問題,但問題是我看書的時候就會很麻煩了。分離成兩個明確命名的函數,也許會好很多。

3. ShowWindow。恩就是它。看到這個名字,不熟悉MFC的人應該都會覺得它是用來顯示一個窗口的。但...,其實它兼有顯示和隱藏窗口的功能,決定這一切的又是一個詭異的bool量...

4. MoveWindow和SetWindowPos。這一對不夠專注的兄弟。它們太強大了,所能做的事遠不止Move和SetPos那么簡單,當我想改變一個窗體大小的時候,我翻來覆去的找函數,萬萬沒想到,原來是這對名不副實的兄弟的工作,Faint...

5. GetWindowRect和GetClientRect。其實他們也許沒有問題,有問題的是我,我不喜歡用引用量代替一個貌似應該由返回值完成的東西...

6. SelectObject。為什么你傳入一個新的Object,它返回一個老的Object呢?不止我一個人,很多人都被這個搞暈過,其實我也不明白,一個函數做兩個函數的事,這確實有點熱心過度了吧...

7. afx_msg。一個永遠都沒被用到的預留量,這樣的東西在MFC中有很多...



duguguiyu 2007-06-21 23:18 發表評論

文章來源:http://www.cnblogs.com/duguguiyu/archive/2007/06/21/792426.html
posted @ 2007-06-21 23:18 duguguiyu 閱讀(236) | 評論 (0)編輯 收藏

利用MFC向導建立一個工程,然后開始編碼。這就是我通常做一個MFC工程的開始。但向導可不是一個守規矩的東西,它會為你添加很多的代碼,為你設置大量的編譯和鏈接選項。大部分時候這種工作是善意的,但是好心不一定辦好事,你不好好了解它,它會給你帶來很多的麻煩。

在配置一個基于OpenCasCade的程序中,我就遇到了很多麻煩。MFC向導在它所生成的View, Document等架構類中都添加了一段如下代碼:

#ifdef _DEBUG
#define new DEBUG_NEW
#endif

在Debug狀態下(VS會為你默認添加一個_DEBUG的預編譯項),你在該類中調用的new操作符都會被DEBUG_NEW所取代,請警惕這個行為,如果你重載過某個類的new,很可能就會由于它導致無法編譯通過或運行不正確。

除此之外一些默認的設置也要注意,在VS2005中是默認支持Unicode的,它會在你的編譯選項中加入/D "_UNICODE" /D "UNICODE"。這就會使得CString和你可能用到的std::string存在很麻煩的轉換問題。你需要修改項目屬性中General-->Character Set為not set,將其設為ununicode,保證與std::string的一致(當然你還可以運用其他的解決方法滿足你的需求)。

有時候IDE也會“好心辦壞事”,比如在一個解決方案中有兩個工程,你為A添加B的編譯依賴,在A的鏈接選項中就會悄悄加上對B生成的dll的引用。當你某天整理代碼取消了這個依賴的時候,你突然發現莫名的出現了很多link錯誤。不要慌張,在A中添加上B鏈接項就好了,這項工作其實是你必須自己做的,只是你添加了依賴編譯器非常主動的幫你完成了。

也許你看上面的錯誤都很簡單,但如果不小心,也許有天也會像我一樣深陷其中半天爬不出來。總之,在天天用VS2005建MFC工程的時候,提前做好兩件事。一件是通讀一遍系統默認生成的代碼,做到心中有數,每一條莫名其妙的東西都要了解一下它的用途;另一件是在剛開始和改變了工程屬性之后查看一下你的編譯和鏈接命令,搞清楚它做了什么事,有時候命令行雖然難記一點,但確實是一目了然,你可以不必每天用命令行編譯程序,但一定要對這些命令心如明鏡,了如指掌才好。



duguguiyu 2007-06-21 00:20 發表評論

文章來源:http://www.cnblogs.com/duguguiyu/archive/2007/06/21/791161.html
posted @ 2007-06-21 00:20 duguguiyu 閱讀(432) | 評論 (0)編輯 收藏
僅列出標題
共2頁: 1 2 
Welcome to my c++ home...

<2025年9月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

常用鏈接

留言簿(1)

隨筆分類(9)

隨筆檔案(9)

搜索

  •  

積分與排名

  • 積分 - 10738
  • 排名 - 1159

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            欧美成人69av| 亚洲一区二区伦理| 欧美ed2k| 亚洲美女性视频| 99国产精品国产精品毛片| 欧美日韩在线三区| 香蕉成人啪国产精品视频综合网| 亚洲自拍都市欧美小说| 国外成人网址| 亚洲国产电影| 欧美色123| 免费h精品视频在线播放| 欧美电影免费观看网站| 午夜精品久久久久久久| 久久精品视频在线| 亚洲午夜国产一区99re久久| 午夜精彩视频在线观看不卡 | 欧美激情精品久久久久久大尺度| 免费高清在线视频一区·| 宅男66日本亚洲欧美视频| 校园春色国产精品| 一区二区高清视频| 久久精品日韩一区二区三区| 一区二区不卡在线视频 午夜欧美不卡在 | 中文亚洲欧美| 在线观看亚洲视频| 一本到高清视频免费精品| 国产在线高清精品| 亚洲精选成人| 在线看片成人| 亚洲欧美一级二级三级| 亚洲九九爱视频| 久久九九热免费视频| 亚洲在线中文字幕| 欧美精品国产| 欧美.www| 一区在线播放视频| 亚洲无线视频| 在线午夜精品自拍| 欧美18av| 欧美成人午夜激情视频| 国产日韩一区二区| 中文国产成人精品| 日韩视频免费| 久久综合福利| 免费日韩视频| 黑人一区二区| 久久成人免费电影| 午夜精品在线看| 欧美天天在线| 日韩一区二区免费高清| 亚洲美女av电影| 美女日韩在线中文字幕| 美女尤物久久精品| 尤妮丝一区二区裸体视频| 午夜免费在线观看精品视频| 亚洲一区二区在线免费观看| 欧美—级a级欧美特级ar全黄| 欧美激情一区三区| 亚洲黄色有码视频| 麻豆精品网站| 亚洲高清在线| av成人免费在线| 欧美人与性动交cc0o| 99精品免费网| 香蕉成人啪国产精品视频综合网| 国产精品国产三级欧美二区 | 亚洲视屏一区| 欧美在线免费视频| 国产一区亚洲一区| 久久久精品一区| 欧美激情欧美狂野欧美精品| 亚洲国产成人av好男人在线观看| 老司机午夜精品| 亚洲国产专区校园欧美| 一区二区三区四区五区在线 | 欧美国产日韩精品| 99re6热在线精品视频播放速度| 国产精品99久久久久久白浆小说| 欧美午夜精品一区| 先锋资源久久| 亚洲高清视频在线观看| 亚洲视频在线播放| 国产亚洲福利社区一区| 久久久蜜臀国产一区二区| 国产日韩一区二区三区在线播放| 久久精品国产久精国产一老狼| 久久天天躁狠狠躁夜夜av| 亚洲高清成人| 欧美午夜视频网站| 久久成人亚洲| 99在线视频精品| 久久久综合视频| 夜夜爽99久久国产综合精品女不卡 | 91久久精品美女高潮| 亚洲一区中文| 亚洲成人自拍视频| 国产精品你懂的在线| 久久综合久久久久88| 日韩午夜免费| 免费看的黄色欧美网站| 亚洲一区二区在线观看视频| 伊人久久婷婷| 国产精品夜夜嗨| 欧美.com| 久久精品日韩| 中文欧美在线视频| 亚洲国产精品成人| 久久成人国产| 亚洲特色特黄| 日韩视频免费看| 黄网站免费久久| 国产免费一区二区三区香蕉精| 欧美激情在线| 久久最新视频| 欧美在线黄色| 亚洲一区二区在线免费观看| 亚洲日本在线观看| 欧美国产第二页| 麻豆av一区二区三区久久| 亚洲欧美福利一区二区| 99亚洲视频| 亚洲人成网站影音先锋播放| 国产一区视频在线看| 国产欧美一区视频| 欧美午夜寂寞影院| 欧美日韩国产一级片| 男人天堂欧美日韩| 久久久夜精品| 久久久久国产精品一区二区| 亚洲欧美日韩国产综合精品二区| 亚洲日本电影| 日韩午夜在线播放| 日韩一级免费观看| 日韩视频在线观看一区二区| 欧美国产第一页| 亚洲国产精品va在线观看黑人| 欧美不卡视频| 亚洲国产日韩一级| 欧美激情一区二区三区在线视频观看| 蜜桃av噜噜一区| 免费日韩成人| 亚洲国产日韩精品| 亚洲乱码精品一二三四区日韩在线| 亚洲福利视频一区二区| 亚洲人体1000| 国产精品99久久久久久久久| 亚洲无人区一区| 午夜精品久久久久久| 性伦欧美刺激片在线观看| 欧美在线视频网站| 久久网站免费| 欧美日韩激情网| 国产精品成人一区二区艾草| 亚洲一区欧美一区| 午夜亚洲伦理| 久久影视精品| 欧美日韩精品免费观看视一区二区| 欧美日韩dvd在线观看| 欧美性淫爽ww久久久久无| 国产精品入口尤物| 精东粉嫩av免费一区二区三区| 亚洲第一狼人社区| 亚洲一级黄色| 久久久精品欧美丰满| 亚洲成色777777在线观看影院| 亚洲精品美女在线观看| 亚洲小视频在线| 久久人体大胆视频| 欧美日韩成人一区| 国产一区二区高清不卡| 亚洲国产精品传媒在线观看| 一本久道综合久久精品| 久久精品国产一区二区电影| 欧美激情国产精品| 亚洲天堂偷拍| 欧美91视频| 国产一区二区三区网站| 一本一本久久| 麻豆乱码国产一区二区三区| 亚洲精品看片| 久久免费视频网| 国产精品亚洲成人| 日韩视频第一页| 久久蜜桃香蕉精品一区二区三区| 亚洲精品老司机| 久色成人在线| 国产农村妇女精品一二区| 亚洲看片免费| 久久婷婷av| 性欧美长视频| 国产精品久久久久久久电影| 亚洲精品美女久久久久| 久久婷婷国产综合国色天香| 一区电影在线观看| 欧美精品一卡| 亚洲三级免费电影| 狂野欧美激情性xxxx欧美| 午夜一区二区三区不卡视频| 欧美日韩午夜激情|