Моя цель - написать класс (назовем его CProgressDlg
), который можно использовать для отображения диалогового окна с индикатором выполнения, когда для завершения какой-либо операции в основном потоке пользовательского интерфейса требуется, скажем, 1 секунда. Итак, ранее написанный метод:
if(do_work)
{
for(int i = 0; i < a_lot; i++)
{
//Do work...
::Sleep(100); //Use sleep to simulate work
}
}
можно легко настроить примерно так (псевдокод):
if(do_work)
{
CProgressDlg m_progDlg;
for(int i = 0; i < a_lot; i++)
{
//Do work...
::Sleep(100); //Use sleep to simulate work
if(m_progDlg.UpdateWithProgress(i))
{
//User canceled it
break;
}
}
}
Поэтому для его реализации я бы запустил рабочий поток из конструктора CProgressDlg
:
::CreateThread(0, 0, ThreadProcProgressDlg, (LPVOID)0, 0, 0);
А затем из рабочего потока я бы создал немодальное диалоговое окно, которое будет отображать индикатор выполнения и кнопку отмены для пользователя:
DWORD WINAPI ThreadProcProgressDlg(
_In_ LPVOID lpParameter
)
{
//Wait a little
::Sleep(1000);
HMODULE hModule = AfxGetResourceHandle();
ASSERT(hModule);
//Get parent window
//(Can't use main window, as its UI thread is blocked)
HWND hParentWnd = NULL;
const static BYTE dlgTemplate[224] = {
0x1, 0x0, 0xff, 0xff, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc8, 0x0, 0xc8, 0x90, 0x4, 0x0, 0x0, 0x0, 0x0, 0x0, 0xdb, 0x0, 0x4b, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x8, 0x0, 0x90, 0x1, 0x0, 0x1, 0x4d, 0x0, 0x53, 0x0, 0x20, 0x0, 0x53, 0x0, 0x68, 0x0, 0x65, 0x0, 0x6c, 0x0, 0x6c, 0x0, 0x20, 0x0, 0x44, 0x0, 0x6c, 0x0, 0x67, 0x0, 0x0, 0x0,
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0x0, 0x1, 0x50, 0x92, 0x0, 0x36, 0x0, 0x42, 0x0, 0xe, 0x0, 0x2, 0x0, 0x0, 0x0, 0xff, 0xff, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x81, 0x0, 0x2, 0x50, 0x7, 0x0, 0x7, 0x0, 0xcd, 0x0, 0x19, 0x0, 0xed, 0x3, 0x0, 0x0, 0xff, 0xff, 0x82, 0x0, 0x0, 0x0, 0x0, 0x0,
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x80, 0x50, 0x7, 0x0, 0x21, 0x0, 0xcd, 0x0, 0x7, 0x0, 0xec, 0x3, 0x0, 0x0, 0x6d, 0x0, 0x73, 0x0, 0x63, 0x0, 0x74, 0x0, 0x6c, 0x0, 0x73, 0x0, 0x5f, 0x0, 0x70, 0x0, 0x72, 0x0, 0x6f, 0x0, 0x67, 0x0, 0x72, 0x0, 0x65, 0x0, 0x73, 0x0, 0x73, 0x0, 0x33, 0x0, 0x32, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x81, 0x0, 0x2, 0x50, 0x7, 0x0, 0x29, 0x0, 0xcd, 0x0, 0x8, 0x0, 0xee, 0x3, 0x0, 0x0, 0xff, 0xff, 0x82, 0x0, 0x0, 0x0, 0x0, 0x0, };
//Show dialog
HWND hDlgWnd = ::CreateDialogIndirectParam(hModule, (LPCDLGTEMPLATE)dlgTemplate, hParentWnd, DlgWndProc, (LPARAM)0);
ASSERT(hDlgWnd);
if(hDlgWnd)
{
::ShowWindow(hDlgWnd, SW_SHOW);
}
return 0;
}
Где минимальная диалоговая процедура (только для ее отображения) будет выглядеть примерно так:
INT_PTR CALLBACK DlgWndProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(wParam);
UNREFERENCED_PARAMETER(lParam);
switch (uMsg)
{
case WM_INITDIALOG:
{
}
return TRUE;
case WM_COMMAND:
{
UINT uCmd = LOWORD(wParam);
if (uCmd == IDOK ||
uCmd == IDCANCEL)
{
::DestroyWindow(hDlg);
return (INT_PTR)TRUE;
}
}
break;
}
return (INT_PTR)FALSE;
}
Но когда я запускаю этот код, мое немодальное диалоговое окно показывается на долю секунды, а затем исчезает. Я понимаю, что я, вероятно, не сделал что-то, чтобы правильно отобразить это из рабочего потока.
Есть идеи, что мне не хватает?