Ending DoModal loop when owner window is destroyed

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?
postrowskiAsked:
Who is Participating?
I wear a lot of hats...

"The solutions and answers provided on Experts Exchange have been extremely helpful to me over the last few years. I wear a lot of hats - Developer, Database Administrator, Help Desk, etc., so I know a lot of things but not a lot about one thing. Experts Exchange gives me answers from people who do know a lot about one thing, in a easy to use platform." -Todd S.

RONSLOWCommented:
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.

0

Experts Exchange Solution brought to you by

Your issues matter to us.

Facing a tech roadblock? Get the help and guidance you need from experienced professionals who care. Ask your question anytime, anywhere, with no hassle.

Start your 7-day free trial
postrowskiAuthor 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.
}
0
RONSLOWCommented:
Looks like all you needs was a push in the right direction.

Glad to have helped

0
It's more than this solution.Get answers and train to solve all your tech problems - anytime, anywhere.Try it for free Edge Out The Competitionfor your dream job with proven skills and certifications.Get started today Stand Outas the employee with proven skills.Start learning today for free Move Your Career Forwardwith certification training in the latest technologies.Start your trial today
System Programming

From novice to tech pro — start learning today.