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

渴望飛翔
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>
            国产精品成人在线| 欧美激情一区二区| 久久久亚洲精品一区二区三区| 影音先锋亚洲精品| 136国产福利精品导航| 国产综合亚洲精品一区二| 国产亚洲欧美日韩精品| 国内精品伊人久久久久av影院| 黄色成人片子| 伊人成人网在线看| 亚洲国产女人aaa毛片在线| 亚洲第一在线综合在线| 亚洲激情视频| 欧美一区二粉嫩精品国产一线天| 美女尤物久久精品| 欧美精品啪啪| 国产一区二区福利| 亚洲精品资源| 久久视频免费观看| 欧美日韩国产精品| 亚洲第一精品夜夜躁人人爽| 亚洲免费不卡| 亚洲欧洲美洲综合色网| 欧美刺激午夜性久久久久久久| 99热这里只有成人精品国产| 久久久人人人| 久久久久久久尹人综合网亚洲| 国产精品免费看| 欧美高清视频免费观看| 亚洲天堂免费观看| 一区二区三区免费观看| 日韩午夜av在线| 亚洲欧美日韩系列| 亚洲综合精品四区| 国产精品看片你懂得| 亚洲久色影视| 久久黄色影院| 亚洲精品久久7777| 美国成人直播| 欧美一区二区三区四区在线观看地址| 久久精品视频一| 久久精品视频在线播放| 激情一区二区| **性色生活片久久毛片| 欧美日韩国产一区二区三区地区| 欧美一区二区免费观在线| 在线一区欧美| 新狼窝色av性久久久久久| 久久久爽爽爽美女图片| 久久久精品国产99久久精品芒果| 久久精品国产免费| 久久综合导航| 久久精品视频免费| 久久精品一区二区三区不卡| 亚洲乱码精品一二三四区日韩在线 | 亚洲精品久久久久久久久久久久久 | 亚洲欧美www| 欧美日本国产在线| 国内精品久久久久国产盗摄免费观看完整版| 久久久国产精品一区二区中文| 亚洲卡通欧美制服中文| 欧美日韩网址| 毛片一区二区三区| 欧美日韩国产成人在线观看| 欧美成人午夜免费视在线看片| 久久福利毛片| 亚洲特级毛片| 欧美精品久久久久久久| 久久午夜影视| 老司机aⅴ在线精品导航| 欧美一区二区三区在线观看视频 | 亚洲天堂av在线免费观看| 欧美一区二区精品久久911| 亚洲精品自在久久| 亚洲日本激情| 欧美高清一区| 99ri日韩精品视频| 亚洲精品网址在线观看| 久久精品国产第一区二区三区最新章节 | 欧美日韩在线播| 亚洲国产天堂久久国产91| 一区在线视频| 欧美在现视频| 欧美大秀在线观看| 亚洲乱码国产乱码精品精| 久久国产精品久久久| 另类激情亚洲| 夜夜嗨av一区二区三区中文字幕| 欧美精品一区在线观看| 夜夜精品视频一区二区| 新片速递亚洲合集欧美合集| 国产午夜亚洲精品不卡| 久久久久久久尹人综合网亚洲| 免费不卡在线视频| 一区二区三区免费观看| 国产精品午夜在线| 久久久久久久久久久久久女国产乱| 久久久综合视频| 99精品视频免费观看视频| 国产亚洲欧美日韩在线一区| 欧美大片在线看| 亚洲欧美一区二区视频| 午夜精品久久久久久久男人的天堂| 国产精品久久久久久福利一牛影视| 欧美一区2区三区4区公司二百| 最新高清无码专区| 久久夜色精品国产| 欧美一区二区三区精品| 亚洲免费高清视频| 亚洲欧洲一区二区三区在线观看 | 久久超碰97人人做人人爱| 欧美福利视频在线观看| 免费亚洲电影在线观看| 欧美在线看片| 久久最新视频| 久久久.com| 久久久91精品国产一区二区精品| 亚洲视频专区在线| 中文欧美字幕免费| 午夜欧美大尺度福利影院在线看| 99精品国产在热久久| 亚洲精选在线| 亚洲欧美清纯在线制服| 午夜在线一区| 葵司免费一区二区三区四区五区| 久久精品一级爱片| 欧美成人有码| 国产精品免费一区二区三区观看| 亚洲精美视频| 亚洲欧美综合v| 亚洲男女自偷自拍| 欧美在线网址| 欧美日韩精选| 在线观看亚洲精品| 欧美在线免费观看视频| 老司机午夜免费精品视频 | 一个色综合av| 欧美18av| 国产日韩精品一区二区| 亚洲福利电影| 欧美一级一区| 亚洲精品久久| 免费观看不卡av| 国产在线精品自拍| 中文久久精品| 欧美福利视频一区| 久久久av网站| 国产欧美日韩| 性欧美超级视频| 亚洲精品欧美激情| 欧美成人r级一区二区三区| 国产欧美一级| 欧美一级二级三级蜜桃| 99re6热只有精品免费观看| 乱中年女人伦av一区二区| 激情一区二区三区| 久久久久久久性| 欧美尤物一区| 亚洲第一中文字幕在线观看| 久久亚洲春色中文字幕| 欧美一级在线视频| 精品福利免费观看| 蜜桃久久精品一区二区| 老司机精品视频网站| 好看不卡的中文字幕| 免费国产一区二区| 久久伊人免费视频| 日韩午夜在线观看视频| 日韩视频免费在线观看| 国产伦精品一区二区三区照片91 | 久久成人亚洲| 噜噜噜91成人网| 亚洲综合国产激情另类一区| 香蕉久久夜色精品国产| 亚洲麻豆一区| 免费短视频成人日韩| 亚洲综合色噜噜狠狠| 久久九九精品99国产精品| 一区在线观看视频| 亚洲一二三级电影| 久久尤物视频| 能在线观看的日韩av| 国产精品视频一区二区高潮| 欧美激情精品久久久六区热门| 国产麻豆精品久久一二三| 欧美黄色一级视频| 亚洲国产成人在线播放| 亚洲直播在线一区| 亚洲视频在线一区观看| 欧美亚洲第一页| 在线综合亚洲欧美在线视频| 亚洲欧洲日韩综合二区| 久久精品人人做人人综合| 久久久久久亚洲精品杨幂换脸| 国产精品看片资源| 午夜视频在线观看一区| 久久久久久尹人网香蕉| 国产精品资源| 久久免费视频在线观看| 麻豆成人91精品二区三区|