Версия для печати темы
Нажмите сюда для просмотра этой темы в оригинальном формате
Форум программистов > Visual C++/MFC/WTL > [CDialog] OnActivate и OnShowWindow не срабатывают


Автор: ksili 12.2.2009, 12:12
Мне надо, чтобы при открытии диалога выполнялись некоторые действия. 
Я добавил для него обработчики события OnActivate и OnShowWindow, поставил в них брекпойнты, они активны. Но при открытии диалога управление ни в один из этих обработчиков не попадает. 
Однако в OnShowWindow управление попадает, когда я после открытия этого диалога открываю какой-нибудь другой диалог. 
В чем может быть дело? Вот кратко код:

*.h
Код

    afx_msg void OnShowWindow(BOOL bShow, UINT nStatus);
    afx_msg void OnActivate(UINT nState, CWnd* pWndOther, BOOL bMinimized);

*.cpp
Код

BEGIN_MESSAGE_MAP(CUpdateFWDlg, CBaseFWDlg)
    ON_WM_ACTIVATE()
    ON_WM_SHOWWINDOW()
END_MESSAGE_MAP()

....................


void CUpdateFWDlg::OnActivate(UINT nState, CWnd* pWndOther, BOOL bMinimized)
{
.......
}

void CUpdateFWDlg::OnShowWindow(BOOL bShow, UINT nStatus)
{
.........
}


Добавлено через 9 минут и 6 секунд
А сейчас управление в OnShowWindow поймалось, когда нажал на одну из кнопок в диалоге. Думаю OnShowWindow вообще не надо использовать - для моего случая она не подходит.

Автор: voov 12.2.2009, 12:25
OnInitDialog не подходит? 

Автор: ksili 12.2.2009, 12:32
Цитата(voov @  12.2.2009,  16:25 Найти цитируемый пост)
OnInitDialog не подходит?

А когда оно срабатывает?

Сейчас OnSetFocus попробую. Если не подойдет, попробую OnInitDialog

Добавлено через 12 минут и 49 секунд
OnSetFocus тоже не сработал.

Я бы рад использовать OnInitDialog, но не пойму как. Сообщения WM_INITDIALOG я не нашёл, есть только WM_INITMENU и WM_INITMENUPOPUP. 
Я посмотрел другой диалог, где уже есть такая функция, но не врубился - есть объявление и определение этой функции, но нигде нет ее вызова. В карте сообщений она тоже не прописана. При этом я проверил: управление в нее попадает только при создании приложения, а не при открытии этого диалога.

Автор: voov 12.2.2009, 12:46
Срабатывает после создания диалога, перед его отображением. Но только 1 раз, если тебе нужно выполнять действия после сварачивания/разварачивания диалога или еще как, то не подойдет.

Автор: Andrey44 12.2.2009, 15:39
ksili, что-то не понятно, что тебе конкретно надо?

Автор: ksili 13.2.2009, 06:38
Цитата(ksili @  12.2.2009,  16:12 Найти цитируемый пост)
Мне надо, чтобы при открытии диалога выполнялись некоторые действия. 

Без всякого ожидания, без нажатия каких-либо кнопок. Сразу.

Автор: Andrey44 13.2.2009, 08:16
ksili, Открытие это когда ты собираешься его показать?
И каждый раз при показе ты хочешь что-то делать?
Ну ты-же его как-то показываешь? Вот и делай что нибудь перед тем как показываешь.

Автор: Earnest 13.2.2009, 08:48
OnInitDialog - это виртуальная функция, нужно ее просто переопределить.
Не знаю, где искал WM_INITDIALOG, если не нашел... видимо в предложениях мастера... smile 
НО: OnInitDialog приходит, когда диалог еще не виден! Если тебя это устраивает, то нормально. А если нет - тогда лови первый WM_KICKIDLE (если диалог модальный). Это спец сообщение MFC (нужно подключить afxpriv.h) которое приходит каждый раз, когда модальный диалог входит в цикл простоя; в первый раз оно приходит после завершения всех инициализаций и отображения диалога.

Автор: ksili 13.2.2009, 08:59
Это диалог проверки обновлений.

Когда он открывается программа проверяет наличие обновлений, в это время отображается надпись Checking for updates...
Затем, в зависимости от результатов проверки, могут отображаться разные надписи, а первая надпись исчезает.

Т.е. делать это надо не до открытия диалога, а сразу после того, как он отобразился на экране.

Автор: Andrey44 13.2.2009, 09:28
Цитата

The framework calls this member function when the nonclient area needs to be changed to indicate an active or inactive state.

 
afx_msg BOOL OnNcActivate(
   BOOL bActive 
);
 


Попробуй может поможет.

Автор: ksili 13.2.2009, 09:56
Цитата(Andrey44 @  13.2.2009,  13:28 Найти цитируемый пост)
Попробуй может поможет.

Тоже не помогло.

Цитата(Earnest @  13.2.2009,  12:48 Найти цитируемый пост)
Не знаю, где искал WM_INITDIALOG, если не нашел... видимо в предложениях мастера...

В который раз убеждаюсь, что слово vusual в названии студии это просто замануха. Почти ничего в ней визуального нету, все руками делать. В Билдере бы проблемы данного топика вообще не возникло - заполнил обработчик формы OnShow, и все.

Автор: Earnest 13.2.2009, 10:02
Не руками, а головой.
Так в чем проблема, программируй в Борланде. А visual означает всего лишь, что сразу видно кто есть кто... smile 

Автор: ksili 13.2.2009, 10:11
Цитата(Earnest @  13.2.2009,  14:02 Найти цитируемый пост)
Так в чем проблема, программируй в Борланде

это зависит не от меня в данном случае

Цитата(Earnest @  13.2.2009,  12:48 Найти цитируемый пост)
WM_KICKIDLE

Интересно, в MSDN отдельной статьи про это сообщение нету. Оно только упоминается в статье про CWnd::RunModalLoop. Но из нее непонятно когда оно приходит
Из ваших слов, Earnest, я понял, что оно будет приходить не только при открытии диалога, а еще и когда он просто открыт, и какое-то время ничего не происходит. Так?

Автор: Earnest 13.2.2009, 10:18
Да, оно приходит каждый раз после опустошения очереди событий. Сразу, а не через некоторое время.

Автор: ksili 13.2.2009, 10:23
Но ведь это же не то, что мне нужно. Мне нужно один раз после открытия диалога (после того как он отобразился на экране)

Автор: Earnest 13.2.2009, 10:26
А флаг поставить, а потом сбросить? В чем проблема-то!

Автор: ksili 13.2.2009, 10:36
Никаких проблем.
Я пытался найти простейшее решение для банальной задачи, а в результате получил предложение использовать какое-то нестандартное сообщение, для которого ещё нужно что-то подключать, и дополнительный флаг.

WM_SHOWWINDOW, которое бы было здесь наиболее логично, вообще срабатывает тогда, когда окно скрывается, а не появляется. Я в шоке.

Вроде бы visual studio, вроде бы MFC, а не голый C++... а все приходится делать через одно место.

Добавлено через 3 минуты и 1 секунду
Цитата(Earnest @  13.2.2009,  14:26 Найти цитируемый пост)
А флаг поставить, а потом сбросить?

кстати а когда его сбрасывать? надо будет ловить момент закрытия диалога? Т.е. использовать что-то типа WM_CLOSE, который конечно же не будет срабатывать когда надо... ах да! у нас же есть на этот случай WM_SHOWWINDOW! гениально

Автор: voov 13.2.2009, 12:29
ksili, не кипятись. прежде, чем биться в истерике, немешало бы разобраться. а претензии типа - "здесь не так как я привык", не обоснованы.

Во-первых, WM_SHOWWINDOW приходит и когда окно показывается и когда окно прячется
Цитата

The WM_SHOWWINDOW message is sent to a window when the window is about to be hidden or shown

OnShowWindow(BOOL bShow, UINT nStatus)
а в обработчике первый параметр как раз и указывает почему в данный момент пришло сообщение.

Во-вторых, OnInitDialog вполне заменяет OnShow в делфи. А то что ты его не нашел среди сообщений в визарде, так это потому что не там искал. OnInitDialog это виртуальная функция, а не сообщение.

Проблема в другом. Ты хочешь чтобы окно открылось, чтото написало, затем занялось своей работой, а потом поменяло надпись. Для этого либо нужно проверять обновления в отдельном потоке, либо сделать как говорила Earnest. В первом случае окно будет реагировать на другие сообщения, в том числе и на нажатие по кнопке ОТМЕНА, а во втором - нет. Выбирай что больше подходит. Кстати, я совсем не уверен, что если это все сделать на делфи в OnShow, то будет работать так, как ты хочешь.


Цитата(Earnest @  13.2.2009,  10:26 Найти цитируемый пост)
А флаг поставить, а потом сбросить? В чем проблема-то!


Цитата(ksili @  13.2.2009,  10:36 Найти цитируемый пост)
кстати а когда его сбрасывать?


ksili, при создании диалога взводишь флаг, при обработке WM_KICKIDLE проверяешь флаг и если он взведен, то выполняешь свои действия, а затем сбрасываешь флаг. При всех последующих обработках WM_KICKIDLE твои действия выполняться не будут, так как флаг уже сброшен.

Автор: ksili 13.2.2009, 12:54
Я пробовал OnInitDialog. Срабатывает один раз при создании приложения.

OnShowWindow разумеется тоже пробовал. Когда оно срабатывает, я уже писал.
Цитата(voov @  13.2.2009,  16:29 Найти цитируемый пост)
ksili, при создании диалога взводишь флаг, при обработке WM_KICKIDLE проверяешь флаг и если он взведен, то выполняешь свои действия, а затем сбрасываешь флаг. При всех последующих обработках WM_KICKIDLE твои действия выполняться не будут, так как флаг уже сброшен.

так будет срабатывать только при первом открытии диалога. А мне надо при каждом

Автор: Earnest 13.2.2009, 18:48
Цитата(ksili @  13.2.2009,  13:54 Найти цитируемый пост)
так будет срабатывать только при первом открытии диалога. А мне надо при каждом 

Так ты что, глобальный флаг собрался поставить? В классе диалога ставь. А диалоги обычно убивают после завершения DoModal. Кстати, он у тебя модальный? 

И совершенно непонятно, чего это у тебя не приходит ShowWindow на открытие окна. Прекрасно оно приходит... скорее всего, что-то неправильно сделал с обработчиком...

Автор: ksili 18.2.2009, 09:20
Наконец-то разобрался!

Диалог активируется функцией SetWindowPos, а не ShowWindow. Поэтому WM_SHOWWINDOW не приходит, вместо него надо ловить WM_WINDOWPOSCHANGED.

Powered by Invision Power Board (http://www.invisionboard.com)
© Invision Power Services (http://www.invisionpower.com)