友情提示:如果本网页打开太慢或显示不完整,请尝试鼠标右键“刷新”本网页!
深入浅出MFC第2版(PDF格式)-第167部分
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部! 如果本书没有阅读完,想下次继续接着阅读,可使用上方 "收藏到我的浏览器" 功能 和 "加入书签" 功能!
這是個隨時等待接受訊息的小程式。它所接受的訊息,夾帶著 global atom 作為參數。
Tracewin 只要把 atom 解碼,丟到個由它管轄的Edit 視窗即可。Paul DiLascia 設計
的這個小程式,功能面面俱到,可以把接受的 TRACE 輸出字串顯示在視窗,或放到
某個檔案;也可以把 EDIT 緩衝區內容拷貝到剪貼簿,或是清除整個 EDIT 緩衝區
內容。功能與 Visual C++ 1。5 的DBWIN 幾乎不相,只差洠軌虬殉e字串輸出到
1 和 2 。
Tracewin 並不動用 Document/View 架構。主視窗內含個 Edit 控制元件作為其子視
窗,事實,那是個衍生自 CEdit 的 CBufWnd 類別,有點像 CEditView 。
當應用程式以 TRACE 巨集送出字串,經過 CDumpContext::OutputString 的作用,送往
CMfxTrace::Write FindWindow Tracewin
,我們在其以 找到 視窗:
CWnd *pTraceWnd = CWnd::FindWindow(TRACEWND_CLASSNAME; NULL);
要知道,如果 Tracewin 使用的類別是 MFC 預先建立的個類別,那麼它的類別名稱
可能是像 Afx:b:14ae:6:3e8f 這種不太有意義的字串,而且可能在不同的機器不同的時間
有不同的名稱。如此來如何為 FindWindow 指定第個參數?我們必須有個什麼方法
避免使用 MFC 預先建立的個類別,但又能夠使用其類別設定。
視窗類別名稱 Afx:x:y:z:w
MFC 2。5 在應用程式開始執行時,便在 AfxWinInit 先行裕粤耍祩視窗類別備用。
MFC 4。x 的行為稍有修改,它在應用程式呼叫 LoadFrame 準備產生視窗時,才在
LoadFrame 所呼叫的 PreCreateWindow 虛擬函式為你產生視窗類別。5個可能的視窗
類別分別是:
928
…………………………………………………………Page 991……………………………………………………………
附錄D 以MFC 重建DBWIN
const TCHAR _afxWnd'' = AFX_WND;
const TCHAR _afxWndControlBar'' = AFX_WNDCONTROLBAR;
const TCHAR _afxWndMDIFrame'' = AFX_WNDMDIFRAME;
const TCHAR _afxWndFrameOrView'' = AFX_WNDFRAMEORVIEW;
const TCHAR _afxWndOleControl'' = AFX_WNDOLECONTROL;
這些 AFX_xxx 常數定義於 AFXIMPL。H ,依不同的聯結狀態(除錯模式與否、動態
聯結與否)而有不同的值。如果使用 MFC 動態聯結版和除錯版,5個視窗類別的名稱
將是:
〃AfxWnd42d〃
〃AfxControlBar42d〃
〃AfxMDIFrame42d〃
〃AfxFrameOrView42d〃
〃AfxOleControl42d〃
如果是使用 MFC 靜態聯結版和除錯版,5個視窗類別的名稱將是:
〃AfxWnd42sd〃
〃AfxControlBar42sd〃
〃AfxMDIFrame42sd〃
〃AfxFrameOrView42sd〃
〃AfxOleControl42sd〃
深入湷錾钊霚出 MFC /
這些名稱的由來,以及它們的裕詴r機,請參考深入湷錾钊霚出 (侯俊傑 松崗)第6
章。
然而,這些視窗類別名稱又怎麼會變成像 Afx:b:14ae:6:3e8f 這副奇怪模樣呢?原來是
Framework 玩的把戲,它把這些視窗類別名稱轉換為 Afx:x:y:z:w 型式,成為獨無之
視窗類別名稱:
x: 視窗風格(window style)的 hex 值
y: 滑鼠游標的 hex 值
z: 背景顏色的 hex 值
w: 圖示的 hex 值
CMfxTrace FindWindow Tracewin Tracewin
為了讓 能夠以 函式找到 , 的視窗類別名稱
Afx:x:y:z:w
(也就是那個 )必須完全在我們的控制之才行。那麼,我們勢必得改寫
Tracewin 的 frame 視窗的 PreCreateWindow 虛擬函式。
929
…………………………………………………………Page 992……………………………………………………………
第五篇 附錄
圖五 以 Spy 觀察視窗。請注意每一個視窗類別的名稱都是 Afx:x:y:z:w 型
式。fig5。bmp
PreCreateWindow 和 GetClassInfo
圖六是 Tracewin 的 PreCreateWindow 內容,利用 GetClassInfo 把 MFC 的視窗類別做
出份副本,然後改變其類別名稱以及程式圖示,再重新裕浴J堑模匦略'冊是必要
CMfxTrace Write FindWindow
的。這麼來, :: 呼叫的 就有明確的搜尋目標了。
#0001 BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs)
#0002 {
#0003 static LPCSTR className = NULL;
#0004
#0005 if (!CFrameWnd::PreCreateWindow(cs))
#0006 return FALSE;
#0007
#0008 if (className==NULL) {
#0009 // One…time class registration
#0010 // The only purpose is to make the class name something
#0011 // meaningful instead of 〃Afx:0x4d:27:32:hup1hup:hike!〃
#0012 //
#0013 WNDCLASS wndcls;
#0014 ::GetClassInfo(AfxGetInstanceHandle(); cs。lpszClass; &wndcls);
#0015 wndcls。lpszClassName = TRACEWND_CLASSNAME;
#0016 wndcls。hIcon = AfxGetApp()…》LoadIcon(IDR_MAINFRAME);
#0017 VERIFY(AfxRegisterClass(&wndcls));
#0018 className=TRACEWND_CLASSNAME;
930
…………………………………………………………Page 993……………………………………………………………
附錄D 以MFC 重建DBWIN
#0019 }
#0020 cs。lpszClass = className;
#0021
#0022 // Load window position from profile
#0023 CWinApp *pApp = AfxGetApp();
#0024 cs。x = pApp…》GetProfileInt(PROFILE; 〃x〃; CW_USEDEFAULT);
#0025 cs。y = pApp…》GetProfileInt(PROFILE; 〃y〃; CW_USEDEFAULT);
#0026 cs。cx = pApp…》GetProfileInt(PROFILE; 〃cx〃; CW_USEDEFAULT);
#0027 cs。cy = pApp…》GetProfileInt(PROFILE; 〃cy〃; CW_USEDEFAULT);
#0028
#0029 return TRUE;
#0030 }
圖六 Tracewin 的 frame 視窗的 PreCreateWindow 函式內容
Tracewin 取出字串並顯示
WM TRACE MSG OnTraceMsg
Tracewin _ _
如果 欲接收由除錯端傳來的自定訊息 ,並交由
函式去處理,它就必須在其訊息映射表有所表示:
BEGIN_MESSAGE_MAP(CMainFrame; CFrameWnd)
//{{AFX_MSG_MAP(CMainFrame)
ON_REGISTERED_MESSAGE(WM_TRACE_MSG; OnTraceMsg)
。。。
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
並且設計 OnTraceMsg 函式如圖七。
#0001 LRESULT CMainFrame::OnTraceMsg(WPARAM wParam; LPARAM)
#0002 {
#0003 if (!wParam || m_nOutputWhere==ID_OUTPUT_OFF)
#0004 return 0;
#0005
#0006 char buf'256';
#0007 UINT len = GlobalGetAtomName((ATOM)wParam; buf; sizeof(buf));
#0008
#0009 if (m_nOutputWhere==ID_OUTPUT_TO_WINDOW) {
#0010
#0011 // Convert n to nr for Windows edit control
931
…………………………………………………………Page 994……………………………………………………………
第五篇 附錄
#0012 。。。
#0013 // Append string to contents of trace buffer
#0014 。。。
#0015 } else if (m_nOutputWhere==ID_OUTPUT_TO_FILE) {
#0016 m_file。Write(buf; len);
#0017 }
#0018 return 0;
#0019 }
圖七 CMainFrame::OnTraceMsg 函式內容。
WM_COPYDATA IPC
改用 進行行程通訊( )
Paul DiLascia 的第篇文章發表後,收到許多讀者的來信,指出以 global atom 完成行
WM COPYDATA
_ Paul
程通訊並不是最高明的辦法,可以改用 。於是 從善如流寫了第
篇文章。
WM COPYDATA
_ 是 Win32 的新訊息,可以提供個簡單又方便的方法,把字串送往另
個程式。這正符合 Tracewin 之所需。這個訊息的兩個參數意義如:
wParam = (WPARAM) (HWND) wnd; // handle of sending window
lParam = (LPARAM) (PCOPYDATASTRUCT) pcds; // pointer to structure with data
其 COPYDATASTRUCT 結構定義如:
typedef struct tagCOPYDATASTRUCT { // cds
DWORD dwData; // 隨便你指定任何你需要的額外資訊
DWORD cbData; // 資料長度
PVOID lpData; // 資料指標
} COPYDATASTRUCT;
dwData ID_COPYDATA_TRACEMSG Tracewin
在本例應用被指定為 ; 將檢查這個識別
WM COPYDATA
碼,如果不合格,就忽略該次的 _ 訊息。
Tracewin 新版本的內容我就不再列出了,請直接載其原始碼看個究竟。
932
…………………………………………………………Page 995……………………………………………………………
附錄D 以MFC 重建DBWIN
我的使用經驗
現在讓我來談點我使用 Tracewin 的經驗。
我早就需要在 Visual C++ 使用 DBWIN 了,也早就看到了 Paul DiLascia 的兩篇文
章,但是真正研讀它並使用其成果,是在我撰寫 /OLE/ActiveX 書(我最新的
本書,還在孵化之)的時候。也許當你讀到該書,會感嘆侯俊傑怎麼能夠對OLE container
和 server 之間的交叉動作瞭若指掌。洠в惺颤N,我只是在 container 和 server 之的每
個我感興趣的函式的開始處,利用 TRACE 輸出些訊息,這樣我就可以從容從
Tracewin 視窗觀察那些函式的被呼喚時機了。
所以我在 OLE container 這麼做:
#include 〃tracewin。h〃
。。。
BOOL CContainerApp::InitInstance()
{
。。。
pMainFrame…》ShowWindow(m_nCmdShow);
pMainFrame…》UpdateWindow();
CMfxTrace::Init(); // add by J。J。Hou
return TRUE;
}
也在 OLE server
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部!
温馨提示: 温看小说的同时发表评论,说出自己的看法和其它小伙伴们分享也不错哦!发表书评还可以获得积分和经验奖励,认真写原创书评 被采纳为精评可以获得大量金币、积分和经验奖励哦!