1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/progdlg.cpp
3 // Purpose: wxProgressDialog
4 // Author: Rickard Westerlund
7 // Copyright: (c) 2010 wxWidgets team
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 // ============================================================================
13 // ============================================================================
15 // ----------------------------------------------------------------------------
17 // ----------------------------------------------------------------------------
19 // For compilers that support precompilation, includes "wx.h".
20 #include "wx/wxprec.h"
26 #if wxUSE_PROGRESSDLG && wxUSE_THREADS
28 #include "wx/progdlg.h"
32 #include "wx/msgdlg.h"
33 #include "wx/stopwatch.h"
34 #include "wx/msw/private.h"
37 #include "wx/msw/private/msgdlg.h"
38 #include "wx/evtloop.h"
40 using namespace wxMSWMessageDialog
;
42 #ifdef wxHAS_MSW_TASKDIALOG
44 // ----------------------------------------------------------------------------
46 // ----------------------------------------------------------------------------
51 // Notification values of wxProgressDialogSharedData::m_notifications
52 const int wxSPDD_VALUE_CHANGED
= 0x0001;
53 const int wxSPDD_RANGE_CHANGED
= 0x0002;
54 const int wxSPDD_PBMARQUEE_CHANGED
= 0x0004;
55 const int wxSPDD_TITLE_CHANGED
= 0x0008;
56 const int wxSPDD_MESSAGE_CHANGED
= 0x0010;
57 const int wxSPDD_EXPINFO_CHANGED
= 0x0020;
58 const int wxSPDD_ENABLE_SKIP
= 0x0040;
59 const int wxSPDD_ENABLE_ABORT
= 0x0080;
60 const int wxSPDD_DISABLE_SKIP
= 0x0100;
61 const int wxSPDD_DISABLE_ABORT
= 0x0200;
62 const int wxSPDD_FINISHED
= 0x0400;
63 const int wxSPDD_DESTROYED
= 0x0800;
65 const int Id_SkipBtn
= wxID_HIGHEST
+ 1;
67 } // anonymous namespace
69 // ============================================================================
71 // ============================================================================
73 // Class used to share data between the main thread and the task dialog runner.
74 class wxProgressDialogSharedData
77 wxProgressDialogSharedData()
81 m_progressBarMarquee
= false;
87 wxCriticalSection m_cs
;
89 wxWindow
*m_parent
; // Parent window only used to center us over it.
90 HWND m_hwnd
; // Task dialog handler
91 long m_style
; // wxProgressDialog style
96 wxString m_expandedInformation
;
97 wxString m_labelCancel
; // Privately used by callback.
98 unsigned long m_timeStop
;
100 wxProgressDialog::State m_state
;
101 bool m_progressBarMarquee
;
104 // Bit field that indicates fields that have been modified by the
105 // main thread so the task dialog runner knows what to update.
109 // Runner thread that takes care of displaying and updating the
111 class wxProgressDialogTaskRunner
: public wxThread
114 wxProgressDialogTaskRunner()
115 : wxThread(wxTHREAD_JOINABLE
)
118 wxProgressDialogSharedData
* GetSharedDataObject()
119 { return &m_sharedData
; }
122 wxProgressDialogSharedData m_sharedData
;
124 virtual void* Entry();
126 static HRESULT CALLBACK
TaskDialogCallbackProc(HWND hwnd
,
136 // A custom event loop which runs until the state of the dialog becomes
138 class wxProgressDialogModalLoop
: public wxEventLoop
141 wxProgressDialogModalLoop(wxProgressDialogSharedData
& data
)
147 virtual void OnNextIteration()
149 wxCriticalSectionLocker
locker(m_data
.m_cs
);
151 if ( m_data
.m_state
== wxProgressDialog::Dismissed
)
155 wxProgressDialogSharedData
& m_data
;
157 wxDECLARE_NO_COPY_CLASS(wxProgressDialogModalLoop
);
160 // ============================================================================
162 // ============================================================================
164 BOOL CALLBACK
DisplayCloseButton(HWND hwnd
, LPARAM lParam
)
166 wxProgressDialogSharedData
*sharedData
=
167 (wxProgressDialogSharedData
*) lParam
;
169 if ( wxGetWindowText( hwnd
) == sharedData
->m_labelCancel
)
171 sharedData
->m_labelCancel
= _("Close");
172 SendMessage( hwnd
, WM_SETTEXT
, 0,
173 (LPARAM
) sharedData
->m_labelCancel
.wx_str() );
181 void PerformNotificationUpdates(HWND hwnd
,
182 wxProgressDialogSharedData
*sharedData
)
184 // Update the appropriate dialog fields.
185 if ( sharedData
->m_notifications
& wxSPDD_RANGE_CHANGED
)
188 TDM_SET_PROGRESS_BAR_RANGE
,
190 MAKELPARAM(0, sharedData
->m_range
) );
193 if ( sharedData
->m_notifications
& wxSPDD_VALUE_CHANGED
)
196 TDM_SET_PROGRESS_BAR_POS
,
201 if ( sharedData
->m_notifications
& wxSPDD_PBMARQUEE_CHANGED
)
203 BOOL val
= sharedData
->m_progressBarMarquee
? TRUE
: FALSE
;
205 TDM_SET_MARQUEE_PROGRESS_BAR
,
209 TDM_SET_PROGRESS_BAR_MARQUEE
,
214 if ( sharedData
->m_notifications
& wxSPDD_TITLE_CHANGED
)
215 ::SetWindowText( hwnd
, sharedData
->m_title
.wx_str() );
217 if ( sharedData
->m_notifications
& wxSPDD_MESSAGE_CHANGED
)
219 // Split the message in the title string and the rest if it has
222 title
= sharedData
->m_message
,
225 const size_t posNL
= title
.find('\n');
226 if ( posNL
!= wxString::npos
)
228 // There can an extra new line between the first and subsequent
229 // lines to separate them as it looks better with the generic
230 // version -- but in this one, they're already separated by the use
231 // of different dialog elements, so suppress the extra new line.
233 if ( posNL
< title
.length() - 1 && title
[posNL
+ 1] == '\n' )
236 body
.assign(title
, posNL
+ numNLs
, wxString::npos
);
239 else // A single line
241 // Don't use title without the body, this doesn't make sense.
246 TDM_SET_ELEMENT_TEXT
,
247 TDE_MAIN_INSTRUCTION
,
248 (LPARAM
) title
.wx_str() );
251 TDM_SET_ELEMENT_TEXT
,
253 (LPARAM
) body
.wx_str() );
256 if ( sharedData
->m_notifications
& wxSPDD_EXPINFO_CHANGED
)
258 const wxString
& expandedInformation
=
259 sharedData
->m_expandedInformation
;
260 if ( !expandedInformation
.empty() )
263 TDM_SET_ELEMENT_TEXT
,
264 TDE_EXPANDED_INFORMATION
,
265 (LPARAM
) expandedInformation
.wx_str() );
269 if ( sharedData
->m_notifications
& wxSPDD_ENABLE_SKIP
)
270 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, TRUE
);
272 if ( sharedData
->m_notifications
& wxSPDD_ENABLE_ABORT
)
273 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, TRUE
);
275 if ( sharedData
->m_notifications
& wxSPDD_DISABLE_SKIP
)
276 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, FALSE
);
278 if ( sharedData
->m_notifications
& wxSPDD_DISABLE_ABORT
)
279 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, FALSE
);
281 // Is the progress finished?
282 if ( sharedData
->m_notifications
& wxSPDD_FINISHED
)
284 sharedData
->m_state
= wxProgressDialog::Finished
;
286 if ( !(sharedData
->m_style
& wxPD_AUTO_HIDE
) )
288 // Change Cancel into Close and activate the button.
289 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, FALSE
);
290 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, TRUE
);
291 ::EnumChildWindows( hwnd
, DisplayCloseButton
,
292 (LPARAM
) sharedData
);
297 } // anonymous namespace
299 #endif // wxHAS_MSW_TASKDIALOG
301 // ============================================================================
302 // wxProgressDialog implementation
303 // ============================================================================
305 wxProgressDialog::wxProgressDialog( const wxString
& title
,
306 const wxString
& message
,
310 : wxGenericProgressDialog(),
311 m_taskDialogRunner(NULL
),
316 #ifdef wxHAS_MSW_TASKDIALOG
317 if ( HasNativeTaskDialog() )
319 SetParent(GetParentForModalDialog(parent
, GetWindowStyle()));
324 DisableOtherWindows();
328 #endif // wxHAS_MSW_TASKDIALOG
330 Create(title
, message
, maximum
, parent
, style
);
333 wxProgressDialog::~wxProgressDialog()
335 #ifdef wxHAS_MSW_TASKDIALOG
336 if ( !m_taskDialogRunner
)
341 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
342 m_sharedData
->m_notifications
|= wxSPDD_DESTROYED
;
345 m_taskDialogRunner
->Wait();
347 delete m_taskDialogRunner
;
349 ReenableOtherWindows();
351 if ( GetTopParent() )
352 GetTopParent()->Raise();
353 #endif // wxHAS_MSW_TASKDIALOG
356 bool wxProgressDialog::Update(int value
, const wxString
& newmsg
, bool *skip
)
358 #ifdef wxHAS_MSW_TASKDIALOG
359 if ( HasNativeTaskDialog() )
362 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
364 // Do nothing in canceled state.
365 if ( !DoNativeBeforeUpdate(skip
) )
370 wxASSERT_MSG( value
<= m_maximum
, wxT("invalid progress value") );
372 m_sharedData
->m_value
= value
;
373 m_sharedData
->m_notifications
|= wxSPDD_VALUE_CHANGED
;
375 if ( !newmsg
.empty() )
378 m_sharedData
->m_message
= newmsg
;
379 m_sharedData
->m_notifications
|= wxSPDD_MESSAGE_CHANGED
;
382 if ( m_sharedData
->m_progressBarMarquee
)
384 m_sharedData
->m_progressBarMarquee
= false;
385 m_sharedData
->m_notifications
|= wxSPDD_PBMARQUEE_CHANGED
;
388 UpdateExpandedInformation( value
);
390 // If we didn't just reach the finish, all we have to do is to
391 // return true if the dialog wasn't cancelled and false otherwise.
392 if ( value
!= m_maximum
|| m_state
== Finished
)
393 return m_sharedData
->m_state
!= Canceled
;
396 // On finishing, the dialog without wxPD_AUTO_HIDE style becomes a
397 // modal one meaning that we must block here until the user
400 m_sharedData
->m_state
= Finished
;
401 m_sharedData
->m_notifications
|= wxSPDD_FINISHED
;
402 if ( HasPDFlag(wxPD_AUTO_HIDE
) )
405 if ( newmsg
.empty() )
407 // Provide the finishing message if the application didn't.
408 m_message
= _("Done.");
409 m_sharedData
->m_message
= m_message
;
410 m_sharedData
->m_notifications
|= wxSPDD_MESSAGE_CHANGED
;
412 } // unlock m_sharedData->m_cs
414 // We only get here when we need to wait for the dialog to terminate so
415 // do just this by running a custom event loop until the dialog is
417 wxProgressDialogModalLoop
loop(*m_sharedData
);
421 #endif // wxHAS_MSW_TASKDIALOG
423 return wxGenericProgressDialog::Update( value
, newmsg
, skip
);
426 bool wxProgressDialog::Pulse(const wxString
& newmsg
, bool *skip
)
428 #ifdef wxHAS_MSW_TASKDIALOG
429 if ( HasNativeTaskDialog() )
431 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
433 // Do nothing in canceled state.
434 if ( !DoNativeBeforeUpdate(skip
) )
437 if ( !m_sharedData
->m_progressBarMarquee
)
439 m_sharedData
->m_progressBarMarquee
= true;
440 m_sharedData
->m_notifications
|= wxSPDD_PBMARQUEE_CHANGED
;
443 if ( !newmsg
.empty() )
446 m_sharedData
->m_message
= newmsg
;
447 m_sharedData
->m_notifications
|= wxSPDD_MESSAGE_CHANGED
;
450 // The value passed here doesn't matter, only elapsed time makes sense
451 // in indeterminate mode anyhow.
452 UpdateExpandedInformation(0);
454 return m_sharedData
->m_state
!= Canceled
;
456 #endif // wxHAS_MSW_TASKDIALOG
458 return wxGenericProgressDialog::Pulse( newmsg
, skip
);
461 bool wxProgressDialog::DoNativeBeforeUpdate(bool *skip
)
463 #ifdef wxHAS_MSW_TASKDIALOG
464 if ( HasNativeTaskDialog() )
466 if ( m_sharedData
->m_skipped
)
468 if ( skip
&& !*skip
)
471 m_sharedData
->m_skipped
= false;
472 m_sharedData
->m_notifications
|= wxSPDD_ENABLE_SKIP
;
476 if ( m_sharedData
->m_state
== Canceled
)
477 m_timeStop
= m_sharedData
->m_timeStop
;
479 return m_sharedData
->m_state
!= Canceled
;
481 #endif // wxHAS_MSW_TASKDIALOG
484 wxFAIL_MSG( "unreachable" );
489 void wxProgressDialog::Resume()
491 wxGenericProgressDialog::Resume();
493 #ifdef wxHAS_MSW_TASKDIALOG
494 if ( HasNativeTaskDialog() )
499 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
500 m_sharedData
->m_state
= m_state
;
502 // "Skip" was disabled when "Cancel" had been clicked, so re-enable
504 m_sharedData
->m_notifications
|= wxSPDD_ENABLE_SKIP
;
506 // Also re-enable "Cancel" itself
507 if ( HasPDFlag(wxPD_CAN_ABORT
) )
508 m_sharedData
->m_notifications
|= wxSPDD_ENABLE_ABORT
;
510 hwnd
= m_sharedData
->m_hwnd
;
511 } // Unlock m_cs, we can't call any function operating on a dialog with
512 // it locked as it can result in a deadlock if the dialog callback is
513 // called by Windows.
515 // After resuming we need to bring the window on top of the Z-order as
516 // it could be hidden by another window shown from the main thread,
517 // e.g. a confirmation dialog asking whether the user really wants to
520 // Notice that this must be done from the main thread as it owns the
521 // currently active window and attempts to do this from the task dialog
522 // thread would simply fail.
523 ::BringWindowToTop(hwnd
);
525 #endif // wxHAS_MSW_TASKDIALOG
528 WXWidget
wxProgressDialog::GetHandle() const
530 #ifdef wxHAS_MSW_TASKDIALOG
531 if ( HasNativeTaskDialog() )
535 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
536 m_sharedData
->m_state
= m_state
;
537 hwnd
= m_sharedData
->m_hwnd
;
542 return wxGenericProgressDialog::GetHandle();
545 int wxProgressDialog::GetValue() const
547 #ifdef wxHAS_MSW_TASKDIALOG
548 if ( HasNativeTaskDialog() )
550 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
551 return m_sharedData
->m_value
;
553 #endif // wxHAS_MSW_TASKDIALOG
555 return wxGenericProgressDialog::GetValue();
558 wxString
wxProgressDialog::GetMessage() const
560 #ifdef wxHAS_MSW_TASKDIALOG
561 if ( HasNativeTaskDialog() )
563 #endif // wxHAS_MSW_TASKDIALOG
565 return wxGenericProgressDialog::GetMessage();
568 void wxProgressDialog::SetRange(int maximum
)
570 #ifdef wxHAS_MSW_TASKDIALOG
571 if ( HasNativeTaskDialog() )
575 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
577 m_sharedData
->m_range
= maximum
;
578 m_sharedData
->m_notifications
|= wxSPDD_RANGE_CHANGED
;
582 #endif // wxHAS_MSW_TASKDIALOG
584 wxGenericProgressDialog::SetRange( maximum
);
587 bool wxProgressDialog::WasSkipped() const
589 #ifdef wxHAS_MSW_TASKDIALOG
590 if ( HasNativeTaskDialog() )
594 // Couldn't be skipped before being shown.
598 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
599 return m_sharedData
->m_skipped
;
601 #endif // wxHAS_MSW_TASKDIALOG
603 return wxGenericProgressDialog::WasSkipped();
606 bool wxProgressDialog::WasCancelled() const
608 #ifdef wxHAS_MSW_TASKDIALOG
609 if ( HasNativeTaskDialog() )
611 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
612 return m_sharedData
->m_state
== Canceled
;
614 #endif // wxHAS_MSW_TASKDIALOG
616 return wxGenericProgressDialog::WasCancelled();
619 void wxProgressDialog::SetTitle(const wxString
& title
)
621 #ifdef wxHAS_MSW_TASKDIALOG
622 if ( HasNativeTaskDialog() )
628 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
629 m_sharedData
->m_title
= title
;
630 m_sharedData
->m_notifications
= wxSPDD_TITLE_CHANGED
;
633 #endif // wxHAS_MSW_TASKDIALOG
635 wxGenericProgressDialog::SetTitle(title
);
638 wxString
wxProgressDialog::GetTitle() const
640 #ifdef wxHAS_MSW_TASKDIALOG
641 if ( HasNativeTaskDialog() )
643 #endif // wxHAS_MSW_TASKDIALOG
645 return wxGenericProgressDialog::GetTitle();
648 bool wxProgressDialog::Show(bool show
)
650 #ifdef wxHAS_MSW_TASKDIALOG
651 if ( HasNativeTaskDialog() )
653 // The dialog can't be hidden at all and showing it again after it had
654 // been shown before doesn't do anything.
655 if ( !show
|| m_taskDialogRunner
)
658 // We're showing the dialog for the first time, create the thread that
660 m_taskDialogRunner
= new wxProgressDialogTaskRunner
;
661 m_sharedData
= m_taskDialogRunner
->GetSharedDataObject();
663 // Initialize shared data.
664 m_sharedData
->m_title
= m_title
;
665 m_sharedData
->m_message
= m_message
;
666 m_sharedData
->m_range
= m_maximum
;
667 m_sharedData
->m_state
= Uncancelable
;
668 m_sharedData
->m_style
= GetPDStyle();
669 m_sharedData
->m_parent
= GetTopParent();
671 if ( HasPDFlag(wxPD_CAN_ABORT
) )
673 m_sharedData
->m_state
= Continue
;
674 m_sharedData
->m_labelCancel
= _("Cancel");
676 else // Dialog can't be cancelled.
678 // We still must have at least a single button in the dialog so
679 // just don't call it "Cancel" in this case.
680 m_sharedData
->m_labelCancel
= _("Close");
683 if ( HasPDFlag(wxPD_ELAPSED_TIME
|
684 wxPD_ESTIMATED_TIME
|
685 wxPD_REMAINING_TIME
) )
687 // Use a non-empty string just to have the collapsible pane shown.
688 m_sharedData
->m_expandedInformation
= " ";
691 // Do launch the thread.
692 if ( m_taskDialogRunner
->Create() != wxTHREAD_NO_ERROR
)
694 wxLogError( "Unable to create thread!" );
698 if ( m_taskDialogRunner
->Run() != wxTHREAD_NO_ERROR
)
700 wxLogError( "Unable to start thread!" );
704 // Do not show the underlying dialog.
707 #endif // wxHAS_MSW_TASKDIALOG
709 return wxGenericProgressDialog::Show( show
);
712 void wxProgressDialog::UpdateExpandedInformation(int value
)
714 #ifdef wxHAS_MSW_TASKDIALOG
715 unsigned long elapsedTime
;
716 unsigned long estimatedTime
;
717 unsigned long remainingTime
;
718 UpdateTimeEstimates(value
, elapsedTime
, estimatedTime
, remainingTime
);
720 int realEstimatedTime
= estimatedTime
,
721 realRemainingTime
= remainingTime
;
722 if ( m_sharedData
->m_progressBarMarquee
)
724 // In indeterminate mode we don't have any estimation neither for the
725 // remaining nor for estimated time.
727 realRemainingTime
= -1;
730 wxString expandedInformation
;
732 // Calculate the three different timing values.
733 if ( HasPDFlag(wxPD_ELAPSED_TIME
) )
735 expandedInformation
<< GetElapsedLabel()
737 << GetFormattedTime(elapsedTime
);
740 if ( HasPDFlag(wxPD_ESTIMATED_TIME
) )
742 if ( !expandedInformation
.empty() )
743 expandedInformation
+= "\n";
745 expandedInformation
<< GetEstimatedLabel()
747 << GetFormattedTime(realEstimatedTime
);
750 if ( HasPDFlag(wxPD_REMAINING_TIME
) )
752 if ( !expandedInformation
.empty() )
753 expandedInformation
+= "\n";
755 expandedInformation
<< GetRemainingLabel()
757 << GetFormattedTime(realRemainingTime
);
760 // Update with new timing information.
761 if ( expandedInformation
!= m_sharedData
->m_expandedInformation
)
763 m_sharedData
->m_expandedInformation
= expandedInformation
;
764 m_sharedData
->m_notifications
|= wxSPDD_EXPINFO_CHANGED
;
766 #else // !wxHAS_MSW_TASKDIALOG
768 #endif // wxHAS_MSW_TASKDIALOG/!wxHAS_MSW_TASKDIALOG
771 // ----------------------------------------------------------------------------
772 // wxProgressDialogTaskRunner and related methods
773 // ----------------------------------------------------------------------------
775 #ifdef wxHAS_MSW_TASKDIALOG
777 void* wxProgressDialogTaskRunner::Entry()
779 WinStruct
<TASKDIALOGCONFIG
> tdc
;
780 wxMSWTaskDialogConfig wxTdc
;
783 wxCriticalSectionLocker
locker(m_sharedData
.m_cs
);
785 wxTdc
.caption
= m_sharedData
.m_title
.wx_str();
786 wxTdc
.message
= m_sharedData
.m_message
.wx_str();
788 // MSWCommonTaskDialogInit() will add an IDCANCEL button but we need to
789 // give it the correct label.
790 wxTdc
.btnOKLabel
= m_sharedData
.m_labelCancel
;
791 wxTdc
.useCustomLabels
= true;
793 wxTdc
.MSWCommonTaskDialogInit( tdc
);
794 tdc
.pfCallback
= TaskDialogCallbackProc
;
795 tdc
.lpCallbackData
= (LONG_PTR
) &m_sharedData
;
797 // Undo some of the effects of MSWCommonTaskDialogInit().
798 tdc
.dwFlags
&= ~TDF_EXPAND_FOOTER_AREA
; // Expand in content area.
799 tdc
.dwCommonButtons
= 0; // Don't use common buttons.
801 if ( m_sharedData
.m_style
& wxPD_CAN_SKIP
)
802 wxTdc
.AddTaskDialogButton( tdc
, Id_SkipBtn
, 0, _("Skip") );
804 tdc
.dwFlags
|= TDF_CALLBACK_TIMER
| TDF_SHOW_PROGRESS_BAR
;
806 if ( !m_sharedData
.m_expandedInformation
.empty() )
808 tdc
.pszExpandedInformation
=
809 m_sharedData
.m_expandedInformation
.wx_str();
813 TaskDialogIndirect_t taskDialogIndirect
= GetTaskDialogIndirectFunc();
814 if ( !taskDialogIndirect
)
818 HRESULT hr
= taskDialogIndirect(&tdc
, &msAns
, NULL
, NULL
);
820 wxLogApiError( "TaskDialogIndirect", hr
);
822 // If the main thread is waiting for us to exit inside the event loop in
823 // Update(), wake it up so that it checks our status again.
831 wxProgressDialogTaskRunner::TaskDialogCallbackProc
836 LPARAM
WXUNUSED(lParam
),
840 wxProgressDialogSharedData
* const sharedData
=
841 (wxProgressDialogSharedData
*) dwRefData
;
843 wxCriticalSectionLocker
locker(sharedData
->m_cs
);
845 switch ( uNotification
)
848 // Store the HWND for the main thread use.
849 sharedData
->m_hwnd
= hwnd
;
851 // Set the maximum value and disable Close button.
853 TDM_SET_PROGRESS_BAR_RANGE
,
855 MAKELPARAM(0, sharedData
->m_range
) );
857 // We always create this task dialog with NULL parent because our
858 // parent in wx sense is a window created from a different thread
859 // and so can't be used as our real parent. However we still center
860 // this window on the parent one as the task dialogs do with their
861 // real parent usually.
862 if ( sharedData
->m_parent
)
864 wxRect
rect(wxRectFromRECT(wxGetWindowRect(hwnd
)));
865 rect
= rect
.CentreIn(sharedData
->m_parent
->GetRect());
878 // If we can't be aborted, the "Close" button will only be enabled
879 // when the progress ends (and not even then with wxPD_AUTO_HIDE).
880 if ( !(sharedData
->m_style
& wxPD_CAN_ABORT
) )
881 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, FALSE
);
884 case TDN_BUTTON_CLICKED
:
888 ::SendMessage(hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, FALSE
);
889 sharedData
->m_skipped
= true;
893 if ( sharedData
->m_state
== wxProgressDialog::Finished
)
895 // If the main thread is waiting for us, tell it that
896 // we're gone (and if it doesn't wait, it's harmless).
897 sharedData
->m_state
= wxProgressDialog::Dismissed
;
899 // Let Windows close the dialog.
903 // Close button on the window triggers an IDCANCEL press,
904 // don't allow it when it should only be possible to close
905 // a finished dialog.
906 if ( sharedData
->m_style
& wxPD_CAN_ABORT
)
910 sharedData
->m_state
== wxProgressDialog::Continue
,
912 "Dialog not in a cancelable state!"
915 ::SendMessage(hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, FALSE
);
916 ::SendMessage(hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, FALSE
);
918 sharedData
->m_timeStop
= wxGetCurrentTime();
919 sharedData
->m_state
= wxProgressDialog::Canceled
;
927 PerformNotificationUpdates(hwnd
, sharedData
);
930 Decide whether we should end the dialog. This is done if either
931 the dialog object itself was destroyed or if the progress
932 finished and we were configured to hide automatically without
933 waiting for the user to dismiss us.
935 Notice that we do not close the dialog if it was cancelled
936 because it's up to the user code in the main thread to decide
937 whether it really wants to cancel the dialog.
939 if ( (sharedData
->m_notifications
& wxSPDD_DESTROYED
) ||
940 (sharedData
->m_state
== wxProgressDialog::Finished
&&
941 sharedData
->m_style
& wxPD_AUTO_HIDE
) )
943 ::EndDialog( hwnd
, IDCLOSE
);
946 sharedData
->m_notifications
= 0;
955 #endif // wxHAS_MSW_TASKDIALOG
957 #endif // wxUSE_PROGRESSDLG && wxUSE_THREADS