总结OnOK()、OnCancel()、OnClose()、OnDestroy()之间的区别(转)

来源:互联网 发布:java classpath级别 编辑:程序博客网 时间:2024/06/05 17:49

 

2009年09月22日 下午 08:33

第一,OnOK()和OnCancel()是CDialog基类的成员函数,而OnClose()和OnDestroy()是CWnd基类的成员函数,即WM消息响应函数。从应用程序结构的角度,拿对话框来说,红色的X对应的是CWnd,而处于对话框中的“确定”、“取消”按钮则对应了CDialog。

第二,OnClose()和OnDestroy()

在单视图程序中,根据<<深入浅出MFC>>所讲,程序退出时执行的操作顺序为(从点X按钮开始)
(1)用户点击X退出按钮,发送了WM_CLOSE消息----->响应OnClose()
(2)在WM_CLOSE消息的处理函数中,调用DestroyWindow()----->销毁与指定CWnd窗口对象关联的窗口,但未销毁CWnd对象
(3)在DestroyWindow()中发送了WM_DESTROY消息----->窗口销毁后响应OnDestroy()
(4)在WM_DESTROY消息中调用PostQuitMessage(),发送WM_QUIT消息,结束消息循环

可以看到,程序的退出过程,是先响应OnClose(),然后响应OnDestroy(),在响应OnDestroy()之前,窗口对象已经被销毁。OnDestroy()到底干了什么呢?它就像一个teller,先通知CWnd对象告诉它即将被销毁,尔后OnDestroy的真正运行是在CWnd对象已经从屏幕上清除以后被调用的。

第三,OnOK()、OnCancel()()、OnClose()、OnDestroy()

CDialog::OnOK首先调用UpdateData(TRUE)将数据传给对话框成员变量,然后调用CDialog::EndDialog关闭对话框;  
CDialog::OnCancel只调用CDialog::EndDialog关闭对话框;  
OnClose()是响应   WM_CLOSE   的.一定程度上可以说CDialog::EndDialog()和OnClose()完成类似的工作,但处理的机制不一样,前者是CDialog的对象机制,后者是WM的消息映射机制。

CDialog::EndDialog()-------->OnDestroy()

                 OnClose()-------->OnDestroy()

EndDialog()和OnClose()属于“同级别”的,所以我们在按下OK按钮的时候,程序是不会执行OnClose()的,但两种机制都必须经过OnDestroy()

 

 

 

然后发现

下面这两种说法不明晰

(1) EndDialog(-1);
关闭模态对话框,并且将参数作为父对话框调用的返回值。
(2)  DestroyWindow();  ::PostQuitMessage(0);
DestroyWindow 关闭非模态对话框。 退出消息循环,真正结束进程。有不少程序窗口关闭,但是不等于退出运行。

 

 

cDialog::onok(),enddialog(),destroywindow区别。

模式和无模式对话的中止是不一样的:模式对话通过调用CDialog : : EndDialog 来中止,无模式对话则是调用CWnd: : DestroyWindow来中止的,函数CDialog : : OnOK和CDialog : : OnCancel调用EndDialog ,所以需要调用DestroyWindow并重置无模式对话的函数。

 

最后三个还不错

Windows API一日一练(18)EndDialog函数 收藏
上一次介绍了怎么样显示对话框的函数,那么怎么样关闭对话框呢?这就需要使用到函数EndDialog。这个函数只能在对话框的消息处理函数里使用,并且这个函数调用之后,没有立即就删除对话框的,而是设置了操作系统里的结束标志。当操作系统查检到有这个标志时,就去删除对话框的消息循环,同时也去释放对话框占用的资源。其实对话框的生命周期是这样的,先由函数DialogBox创建对话框,这样函数DialogBox完成创建对话框但还没有显示前会发出消息WM_INITDIALOG,让对话框有机会初始化上面所有窗口或控件的显示,比如设置文本框的字符串等。最后当用户点出确定或者取消的按钮,就收到两个命令IDOK或IDCANCEL,这时就可以调用函数EndDialog来结束对话框的生命。
函数EndDialog声明如下:
WINUSERAPI
BOOL
WINAPI
EndDialog(
    __in HWND hDlg,
    __in INT_PTR nResult);
hDlg是对话框窗口的句柄。
nResult是设置给函数DialogBox的返回值。
调用这个函数的例子如下:
#001 // 显示关于对话框。
#002 //
#003 // 蔡军生 2007/07/12
#004 //
#005 INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
#006 {
#007  UNREFERENCED_PARAMETER(lParam);
#008  switch (message)
#009  {
#010  case WM_INITDIALOG:
#011         return (INT_PTR)TRUE;
#012
#013  case WM_COMMAND:
#014         if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
#015         {
#016              EndDialog(hDlg, LOWORD(wParam));
#017               return (INT_PTR)TRUE;
#018         }
#019         break;
#020  }
#021  return (INT_PTR)FALSE;
#022 }
第16行就是调用函数EndDialog来关闭对话框。 

 

 

非模式对话框

2009-03-23 17:10

1、非模式对话框的打开:
// 弹出无模式对话框
    CDevPproperty *pDlg= new CDevPproperty;
    pDlg->Create(IDD_DEV_PROPERTY,GetDesktopWindow());
    // 填充设备属性,用SetDlgItemInt()时就不用再定义一个CString来将数据转换成字符串了
    pDlg->SetDlgItemInt(IDC_EDT_DEV_ID,i,TRUE);

    // 为了实现向组合框发送选项,而又不想定义变量,所以这里用了消息传递  
    pWnd=pDlg->GetDlgItem(IDC_CMB_DEV_STATUS);
    pWnd->SetFocus();   // 设置对话框中的焦点
    pWnd->SendMessage(CB_SETCURSEL,CDeviceInfo[i].status,0);

    // 显示窗口
    pDlg->ShowWindow(SW_SHOW);

2、关闭非模式对话框:
DestroyWindow();
delete this;
3、在非模式对话框中向主对话框发送消息:

// 获取全局句柄,然后调用Invalidate()来更新窗口
AfxGetMainWnd()->Invalidate();

4、主窗口中向非模式对话框发送消息

    // 填充设备属性,用SetDlgItemInt()时就不用再定义一个CString来将数据转换成字符串了
    pDlg->SetDlgItemInt(IDC_EDT_DEV_ID,i,TRUE);

    // 为了实现向组合框发送选项,而又不想定义变量,所以这里用了消息传递  
    pWnd=pDlg->GetDlgItem(IDC_CMB_DEV_STATUS);
    pWnd->SetFocus();   // 设置对话框中的焦点
    pWnd->SendMessage(CB_SETCURSEL,CDeviceInfo[i].status,0);
5、将非模式对话框显示在父窗口后面,并且可以切换

一种解决办法是:
建立非模式对话框时Create的第二个参数用GetDesktopWindow(),
m_pDlg->Create(IDD_,GetDesktopWindow());

如果需要恢复Toolbar的属性:
m_pDlg->SetWindowPos(&wndTopMost,0,0,0,0,SWP_NOMOVE|SWP_NOSIZE);

现在有出现了一个问题:系统的任务栏上出现了非模式对话框的图标,好像该对话框和父窗口是两个应用。解决的办法是:
1 定义对象 CWnd *m_pWnd,该对象的父窗口为GetDesktopWindow,设置该对象ShowWindow(SW_HIDE);
2 将非模式对话框的父窗口设置为m_pWnd。

6、非模式对话框与主对话框是一个消息循环

7、如何取得非模式对话框的父窗口指针

取父窗口指针用GetParent()

class CWnd* hWnd = FindWindow(NULL,"窗口标题");

8、基于文档/视图的主窗口均是CMainFrame对象,需要在CView内响应的消息应该这样发送:
CMainFrame *pwnd = (CMainFrame *)GetParent();
pwnd->GetActiveView()->SendMessage(...)

9、怎样才能在线程中实现对话框的顶层显示。

不知为什么设置成WS_EX_TOPMOST并不能实现,可能我们还没有真正理解它该怎样使用。但我用另外的方法实现了:

SetWindowPos (&wndTopMost, 0, 0, 0, 0,SWP_NOMOVE | SWP_NOSIZE)。

10、建立非模态对话框时,它总是在主窗口的最上面,如何才能使它的主窗口显示在上面.
答:1)你有没有试过AfxGetMainWnd()->SetForegroundWindow(),在建立你的非模态对话框之后?
2)当你建立对话框时,向导建立的构造函数有一个指针指出该对话框的父窗口,如果你输入一个窗口,那么该对话框将总是显示在该窗口的上面,如果你输入一个NULL 那么该对话框就可以在主程序窗口的上面或者下面了.不过这时要仔细考虑用户界面,如果非模态对话框在主窗口消失,会不会让你的用户产生误会?是否将非模态对话框显示在任务条上.

11、MSDN中非模式对话框的代码
CMyDialog* pDialog;

void CMyWnd::OnSomeAction()
{
   //pDialog initialized to NULL in the constructor of CMyWnd class
   pDialog = new CMyDialog();
   //Check if new succeeded and we got a valid pointer to a dialog object
   if(pDialog != NULL)
   {
      BOOL ret = pDialog->Create(IDD_MYDIALOG,this);
      if(!ret)   //Create failed.
         AfxMessageBox("Error creating Dialog");
      pDialog->ShowWindow(SW_SHOW);
   }
   else
      AfxMessageBox("Error Creating Dialog Object");
}

14、
由于非模式对话框是在堆中动态分配的,所以每次弹出时,其中的一些变量如果弹出多个的话会有些冲突,比如我在显示每个设备的电量时,由于要用图形显示出来,所以要保存好原来的位置,然后再从第一个位置开始循环画点,这个时候就会发现,弹出的多个对话框中的值是相同的,因此必须要区分开来,我用了两种办法,一是用数组,可以是二维,也可以是一维的,这样呢,用其中的id号作标识。另一个方法是将这些全局变量声明在类内,作为类的变量存在,这样就不会互相干扰了。看代码

在非模式对话框头文件中加入变量:其中m_pt[]是为了保存所有的点的位置,而m_pt_num保存的是点的数目,m_index保存的是一个循环的索引

CPoint m_pt[X_GRID_NUM];
int m_pt_num;
int m_index;

在非模式对话框程序中加入:

CBrush drawBrush;
drawBrush.CreateSolidBrush(RGB(255,255,0));   // 初始化画刷,为黄色
pDC->SelectObject(&drawBrush);       // 选择画刷
pDC->Ellipse(CRect(-3,-3,3,3)); // 画圆,RFD类型为圆圈

Sleep(100);
len = (rc.right-20)/X_GRID_NUM;
m_pt_num = (m_pt_num+1) % X_GRID_NUM;
m_pt[m_pt_num].x=len*m_pt_num;
m_pt[m_pt_num].y=-CDeviceInfo[id].power/2;
for(m_index=1; m_index<=m_pt_num; m_index++)
{
   pDC->Ellipse(CRect(m_pt[m_index].x-1,m_pt[m_index].y-1,m_pt[m_index].x+2,m_pt[m_index].y+2));
    pDC->MoveTo(m_pt[m_index-1]);
    pDC->LineTo(m_pt[m_index]);

 

 

 

这个很棒

对话框默认用的两个按钮的ID分别是IDOK和IDCANCEL,这两个都是在winuser.h 中预定义的系统标准控件ID。 对于标准ID,你不重载时MFC会自动调用父类的相应处理函数。 比如IDOK映射到CDialog::OnOK()函数,IDCANCEL映射到CDialog::OnCancel()。 在这两个函数的源码如下: void CDialog::OnOK() { if (!UpdateData(TRUE)) { TRACE(traceAppMsg, 0, "UpdateData failed during dialog termination./n"); // the UpdateData routine will set focus to correct item return; } EndDialog(IDOK); } void CDialog::OnCancel() { EndDialog(IDCANCEL); } 可以看出点击这两个按钮,都会调用EndDialog()来关闭对话框,只是返回值不同。 EndDialog()函数调用了DestroyWindow()函数,DestroyWindow()函数又发送了WM_DESTROY消息,该消息的处理函数是OnDestroy(),对话框的生存期最后一个函数是PostNcDestroy()函数。 点那个叉叉呢,首先向对话框发送WM_CLOSE消息,由OnClose()函数处理,它调用DestroyWindow(),其后是和上面一样的路由。 可以看出点叉叉的时候绕过了OnOK()和OnCancel()。 小结一下: 1. 点“确定”、“取消”时的关闭路由为 OnOK()或OnCancel() ---> EndDialog() ---> DestroyWindow() ---> OnDestroy() ---> PostNcDestroy() 2. 点“关闭”标题栏按钮的关闭路由为 OnClose()---> DestroyWindow() ---> OnDestroy() ---> PostNcDestroy() 回答楼主的问题: 请注意,上面提到的这些函数统统都是可以重载的,在重载时加入了你自己的代码后,应该调用父类CDialog同名的函数才能正确路由下去,否则就关不了对话框了。 举个例子,重载了关闭的小叉叉 void CAboutDlg::OnClose() { // TODO: 在此添加消息处理程序代码和/或调用默认值 DoSomthing(0; // 执行自己的判断等等 // CDialog::OnClose(); // 把向导生成的父类调用给注释了,这时就关不了对话框了。 } 补充回答,点叉叉会发送WM_CLOSE消息,如果需要重载的话,应该在对话框的属性窗口中,选择WM_CLOSE消息来添加消息处理函数。 VS的IDE会自动添加如下三段: 1. xxx.h文件,类声明中加入OnClose()函数声明 afx_msg void OnClose(); 2. xxx.cpp文件,加入消息映射宏 ON_WM_CLOSE() // 对于Windows标准消息,都是这种简短的格式。 3. xxx.cpp文件,加入函数体 void CMyDlg::OnClose() { CDialog::OnClose(); } 上述3处如果都正常的话,叉叉就映射到OnClose()了。你说的映射到OnCancel()个人觉得有两种可能,第一、缺ON_WM_CLOSE()宏,却多个一个ON_BN_CLICKED(IDCLOSE, &CMyDlg::OnCancel)宏第二、在OnClose()中调用了OnCancel()

原创粉丝点击