We help IT Professionals succeed at work.

We've partnered with Certified Experts, Carl Webster and Richard Faulkner, to bring you two Citrix podcasts. Learn about 2020 trends and get answers to your biggest Citrix questions!Listen Now

x

Ending DoModal loop when owner window is destroyed

postrowski
postrowski asked
on
Medium Priority
1,162 Views
Last Modified: 2013-11-20
I need to close a whole "stack" of windows when a stimulus occurs at the top level. The Windows Owner/Owned relationship will destroy an owned window when the owner is destroyed, but this will not cause EndDialog to run (or, more importantly, EndModalLoop). So, after the window is destroyed, the app is still (at some level of the call stack), running in RunModalLoop and when the app returns there, the call to ContinueModal still returns TRUE (because EndModalLoop has yet to run).

I need to fix this for a number of cases in the application, and it would be cumbersome to add a special handler (say, OnDestroy) to every class or to interpose a special MyAppCDialog class with this handler in the class hierarchy.

What can I do to get EndDialog or EndModalLoop to run?
Comment
Watch Question

Commented:
Loop thru all your child windows (and children of children)

Get a pointer to the CWnd* (derived) object for each one

Check if the pointer IsKindOf CDialog.

If so, call EndDialog for the window.

Not the solution you were looking for? Getting a personalized solution is easy.

Ask the Experts

Author

Commented:
Well, I can't loop through all of the child windows, since an owned window is not a child window, but the IsKindOf function helped out a lot.

What I can do is enumerate all of the top level windows in the system with:

EnumWindows(CMyApp::CloseIfOwned, (LPARAM)MAIN->m_hWnd);

And then check each top level window to see if it is owned by the main application window (or owned by a window that is owned by the main window, etc.). If so, check to see if it IsKindOf(CDialog) or IsKindOf(CPropertySheet) (since CPropertySheet is not descended from CDialog). This approach seems to work just fine. The enumeration callback function is listed below. Thanks for your help.

BOOL CALLBACK CMyApp::CloseIfOwned(
HWND hwnd,// handle to window in question
LPARAM lParam // handle to main app window
)
{
   HWND hOwner;
   HWND hTmp = hwnd;
   do
   {
      hOwner = ::GetWindow(hTmp, GW_OWNER);

      if(hOwner == NULL)
         return TRUE; // always continue enumeration.

      hTmp = hOwner;
   } while (hOwner != (HWND)lParam);

   // Here we know that hwnd is owned by us.

   CWnd *pWnd;
   pWnd = CWnd::FromHandlePermanent(hwnd);
   if(pWnd != NULL)
   {
      if(pWnd->IsKindOf(RUNTIME_CLASS(CDialog)) ||
         pWnd->IsKindOf(RUNTIME_CLASS(CPropertySheet)))
         ((CDialog *)pWnd)->EndDialog(IDCANCEL);
   }

   ::DestroyWindow(hwnd);

   return TRUE; // always continue enumeration.
}

Commented:
Looks like all you needs was a push in the right direction.

Glad to have helped

Access more of Experts Exchange with a free account
Thanks for using Experts Exchange.

Create a free account to continue.

Limited access with a free account allows you to:

  • View three pieces of content (articles, solutions, posts, and videos)
  • Ask the experts questions (counted toward content limit)
  • Customize your dashboard and profile

*This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.

OR

Please enter a first name

Please enter a last name

8+ characters (letters, numbers, and a symbol)

By clicking, you agree to the Terms of Use and Privacy Policy.