浅谈 MFC 的子类化机制和该机制的一个应用(1) (转)

浅谈 MFC 的子类化机制和该机制的一个应用(1) (转)[@more@]

浅谈 MFC 的子类化机制和该机制的一个应用

众所周知:

afx_msg int CWnd::OnCreate( LPCREATESTRUCT lpCreateStruct );XML:namespace prefix = o ns = "urn:schemas-microsoft-com:Office:office" />

是一个经常被重载的 MFC 窗体函数,他负责处理窗体的 WM_CREATE 消息,这个消息的发送时机在窗体刚刚创建以后,CreateWindow(Ex) 返回之前。

 

可以发现在 MFC 里,系统控件和对话框也可以得到这个消息,例如 CEdit,CPrintDialog, CFileDialog,他们内部调用CreateWindowEx,PrintDlg,GetOpen(Save)FileName 完全掩盖了窗体创建的过程,这些函数返回时,窗体已经收到过 WM_CREATE 消息而且不会得到第二次通知。

 

因此,为了得到这些窗体的 WM_CREATE 通知,必须采用有点特殊的方法,能够在CreateWindowEx 返回之前就替换掉窗体的 WindowProc。

 

WH_Cbt 钩子是不错的选择。当一个窗体产生,CBTProc 会在WindowProc 收到 WM_CREATE之前得到 HCBT_CREATEWND 通知,如果此时子类化窗体,就能在子类化后的窗体过程中得到 WM_CREATE 通知。

 

MFC 的做法和这类似:

void AfxHookWindowCreate( CWnd *pWnd );

负责安装 WH_CBT 钩子,其参数 pWnd 指向一个创建中的 CWnd 实例,MFC 通过某种全局变量把这个实例的指针传给执行中的 CBTProc。

BOOL AfxUnhookWindowCreate();

它卸下 WH_CBT 钩子,并且复原 AfxHookWindowCreate 改变过的 MFC 全局状态 消息。

LRESULT CALLBACK _AfxCbtFilterHook(int code, WPARAM wParam, LPARAM lParam);

这个就是 MFC 安装的 CBTProc 回调。他只处理 HCBT_CREATEWND 通知,然后Attach 之前得到的 CWnd 实例到正在创建的窗体句柄、替换掉窗体的 WindowProc,最后 CallNextHookEx。

 

现在来考虑如何应用MFC给我们提供的这个便利。AfxHookWindowCreate 和AfxUnhookWindowCreate 之间创建的第一个非 IME(输入法)窗体可以被所给的 CWnd 实例子类化。我们可以这样调用一些 api,把自己的 CWnd 实例与 API 创建的窗体连结起来,如果我给的是一个 CWnd 派生类的实例,重载过的消息就可以改变原有窗体的行为。以下的代码示例如何按照这样的思路创建一个带 Dump 输出的 MessageBox:

 

ASPectratio="t">

CSDN_Dev_Image_a%20href=2003-7-31420440.png" o:title="DumpMsgBox">

 

class CDumpMsgBox : public CWnd 

{

  DECLARE_DYNAMIC(CDumpMsgBox)

 

// Constructors

public:

  CDumpMsgBox();

 

// Attributes

public:

  CEdit m_editDump;  // Dumping edit control added to the message box.

 

  CMemFile m_fileDump;  // Dumping context's target file.

  CDumpContext m_dumpContext;  // The dump context object.

 

// Operations

public:

  int domessageBox(UINT nIDPrompt, UINT nType = MB_OK, UINT nIDHelp = (UINT)-1);

 

  AFX_INLINE CDumpContext& GetDumpContext()  {  return m_dumpContext; };

  AFX_INLINE void RemoveAll()  {  m_fileDump.SetLength(0);  };

 

// Implementations

public:

  virtual ~CDumpMsgBox();

  virtual void DoDumpObject(CObject* pDumpObject);

  virtual int DoMessageBox(LPCTSTR lpszText, UINT nType = MB_OK, UINT nIDHelp = 0);

 

#ifdef _DEbug

  virtual void AssertValid() const;

  virtual void Dump(CDumpContext& dc) const;

#endif

 

protected:

  virtual BOOL OnDumpOut(LPSTR pszDumpBuffer, UINT nBufferSize);

 

// Overrides

  // ClassWizard generated virtual function overrides

  //{{AFX_VIRTUAL(CChildFrame)

  //}}AFX_VIRTUAL

 

// Generated message map functions

protected:

  //{{AFX_MSG(CMainFrame)

  afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);

  // NOTE - the ClassWizard will add and remove member functions here.

  //  DO NOT EDIT what you see in these blocks of generated code!

  //}}AFX_MSG

  DECLARE_MESSAGE_MAP()

};

 

// Identify of the edit control CDumpMsgBox::m_editDump.

#define IDC_DUMPMSGBOX_EDITBOX  1047

 

//

// CDumpMsgBox

 

IMPLEMENT_DYNAMIC(CDumpMsgBox, CWnd)

 

BEGIN_MESSAGE_MAP(CDumpMsgBox, CWnd)

  //{{AFX_MSG_MAP(CWnd)

  // NOTE - the ClassWizard will add and remove mapping macros here.

  //  DO NOT EDIT what you see in these blocks of generated code !

  ON_WM_CREATE()

  //}}AFX_MSG_MAP

END_MESSAGE_MAP()

 

// CDumpMsgBox construction/destruction

 

CDumpMsgBox::CDumpMsgBox() : m_fileDump(512),

   m_dumpContext(&m_fileDump)

{

  m_fileDump.AssertValid();

 

  m_dumpContext.SetDepth(1);

  m_dumpContext << "Dump From Objects: ======================================== ";

}

 

CDumpMsgBox::~CDumpMsgBox()

{

  BYTE* pszDumpBuffer = (BYTE*)m_fileDump.Detach();

 

  if (pszDumpBuffer)

  free(pszDumpBuffer);

 

  m_dumpContext.m_pFile = NULL;

}

 

// CDumpMsgBox Implementations

 

int CDumpMsgBox::OnCreate(LPCREATESTRUCT lpcs)

{

  if (CWnd::OnCreate(lpcs) == -1)

  {

  TRACE0("CDumpMsgBox::OnCreate error: call CWnd::OnCreate return FALSE. ");

  return -1;

  }

 

  CRect rectBox;

  GetWindowRect(&rectBox);

 

  // adjust message box sizes to fill a dump edit control.

  if (rectBox.Width() < 350)

  rectBox.right = rectBox.left + 350;

  rectBox.bottom += 130;

  MoveWindow(&rectBox, FALSE);

 

  // create edit control to display dump texts

  ScreenToClient(&rectBox);

  CRect rectEdit(rectBox.left + 8, rectBox.bottom - 132,

  rectBox.right - 8, rectBox.bottom - 8);

  if (!m_editDump.Create(WS_CHILD | WS_VISIBLE | WS_VSCROLL | WS_TABSTOP |

  ES_MULTILINE | ES_READONLY | ES_AUTOVSCROLL,

  rectEdit, this, IDC_DUMPMSGBOX_EDITBOX))

  {

  TRACE0("CDumpMsgBox::OnCreate error: m_editDump.Create return FALSE. ");

  return -1;

  }

 

  // set WS_EX_CLIENTEDGE style to edit control (let it has a drop edge)

  m_editDump.ModifyStyleEx(0L, WS_EX_CLIENTEDGE,

  SWP_NOSIZE | SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE | SWP_FRAMECHANGED);

 

  // set edit control's font to DEFAULT_GUI_FONT.

  CFont Font;

  Font.Attach((HFONT)GetStockObject(DEFAULT_GUI_FONT));

  m_editDump.SetFont(&Font);

 

  // write end-dump text and flush all data to file

  m_dumpContext << " ======================================== Dump End";

  m_dumpContext.Flush();

 

  m_fileDump.Write("", sizeof(CHAR));

 

  UINT nBufferSize = m_fileDump.GetLength();

  LPSTR pszDumpBuffer = (LPSTR)m_fileDump.Detach();

 

  if (!OnDumpOut(pszDumpBuffer, nBufferSize))

  {

  TRACE0("CDumpMsgBox::OnCreate error: OnDumpOut return FALSE. ");

 

  m_fileDump.Attach((BYTE*)pszDumpBuffer, nBufferSize, 512);

  return -1;

  }

 

  // attach used dump buffer for next dump.

  m_fileDump.Attach((BYTE*)pszDumpBuffer, nBufferSize, 512);

 

  return 0;

}

 

未完,剩余部分请参考 浅谈 MFC 的子类化机制和该机制的一个应用(2)。

 

 

 


来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/10748419/viewspace-958419/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/10748419/viewspace-958419/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值