最近專案中使用到MFC,由於工程是DLL的,在使用ToolTip時碰到非模態對話方塊無法顯示的問題。查了一番資料,發現原因是由於:雖然MFC Regular DLL派生了CWinApp類,並有一個theApp全域性物件。但它不包含CWinApp::Run機制,主訊息由exe負責接收、分發,導致DLL的PreTranslateMessage不生效。參考資料:https://www.cnblogs.com/hanford/p/6177904.html 第2.5 PreTranslateMessage小節。該文中提到使DLL呼叫PreTranslateMessage的方法,但對於NX二次開發來說無法實現,畢竟主exe是已經封裝好的。又經過一陣搜尋,發現鉤子函式可以解決,具體方法如下:
第一步:在App類中定義鉤子和對話方塊變數
class CTestApp : public CWinApp { public: CTestApp(); CTestDlg* m_dlg; HHOOK m_hHook;
第二步:建構函式中初始化成員變數
CTestApp::CTestApp():m_hHook(NULL),m_dlg(NULL) { // TODO: 在此處新增構造程式碼, // 將所有重要的初始化放置在 InitInstance 中 }
第三步:定義鉤子函式
加粗部分千萬不能省掉,需要判斷訊息是否屬於該對話方塊,否則可能導致介面卡死
class CTestApp : public CWinApp { public: CTestApp(); CTestDlg* m_dlg; HHOOK m_hHook; // 重寫 public: virtual BOOL InitInstance(); DECLARE_MESSAGE_MAP() virtual int ExitInstance(); static LRESULT CALLBACK GetMessageProc(int nCode, WPARAM wParam, LPARAM lParam); };
LRESULT CALLBACK CTestApp::GetMessageProc(int nCode, WPARAM wParam, LPARAM lParam) { AFX_MANAGE_STATE(AfxGetStaticModuleState( )); LPMSG lpMsg = (LPMSG)lParam; if( (AfxGetApp()->PreTranslateMessage(lpMsg)) && (lpMsg->hwnd == theApp.m_dlg->m_hWnd)) { theApp.m_dlg->PreTranslateMessage(lpMsg); } return CallNextHookEx(theApp.m_hHook, nCode, wParam, lParam); }
第四步:安裝鉤子
BOOL CTestApp::InitInstance() { BOOL bInit = CWinApp::InitInstance(); if (bInit) { // TODO: Add your own module initialization code here. m_hHook = ::SetWindowsHookEx( WH_GETMESSAGE, GetMessageProc, AfxGetInstanceHandle(), GetCurrentThreadId()); return CWinApp::InitInstance(); } return bInit; }
第五步:解除安裝鉤子
int CTestApp::ExitInstance() { // TODO: 在此新增專用程式碼和/或呼叫基類 UnhookWindowsHookEx((HHOOK)m_hHook); return CWinApp::ExitInstance(); }
第六步:建立非模態對話方塊
static UF_MB_cb_status_t test( UF_MB_widget_t widget, UF_MB_data_t client_data, UF_MB_activated_button_p_t call_button ) { if (UF_initialize() != 0) return UF_MB_CB_ERROR; AFX_MANAGE_STATE(AfxGetStaticModuleState()); if (theApp.m_dlg == NULL) { theApp.m_dlg = new CTestDlg(); theApp.m_dlg->Create(CTestDlg::IDD); } theApp.m_dlg->ShowWindow(SW_NORMAL); UF_terminate(); return UF_MB_CB_CANCEL; }
大功告成!測試: