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/msw/private/msgdlg.h"
29 #include "wx/progdlg.h"
31 using namespace wxMSWMessageDialog
;
33 #ifdef wxHAS_MSW_TASKDIALOG
35 // ----------------------------------------------------------------------------
37 // ----------------------------------------------------------------------------
42 // Notification values of wxProgressDialogSharedData::m_notifications
43 const int wxSPDD_VALUE_CHANGED
= 0x0001;
44 const int wxSPDD_RANGE_CHANGED
= 0x0002;
45 const int wxSPDD_PBMARQUEE_CHANGED
= 0x0004;
46 const int wxSPDD_TITLE_CHANGED
= 0x0008;
47 const int wxSPDD_MESSAGE_CHANGED
= 0x0010;
48 const int wxSPDD_EXPINFO_CHANGED
= 0x0020;
49 const int wxSPDD_ENABLE_SKIP
= 0x0040;
50 const int wxSPDD_ENABLE_ABORT
= 0x0080;
51 const int wxSPDD_DISABLE_SKIP
= 0x0100;
52 const int wxSPDD_DISABLE_ABORT
= 0x0200;
53 const int wxSPDD_FINISHED
= 0x0400;
54 const int wxSPDD_DESTROYED
= 0x0800;
56 const int Id_SkipBtn
= wxID_HIGHEST
+ 1;
58 } // anonymous namespace
60 // ============================================================================
62 // ============================================================================
64 // Class used to share data between the main thread and the task dialog runner.
65 class wxProgressDialogSharedData
68 wxProgressDialogSharedData()
72 m_progressBarMarquee
= false;
77 wxCriticalSection m_cs
;
79 HWND m_hwnd
; // Task dialog handler
80 long m_style
; // wxProgressDialog style
85 wxString m_expandedInformation
;
86 wxString m_labelCancel
; // Privately used by callback.
87 unsigned long m_timeStop
;
89 wxGenericProgressDialog::ProgressDialogState m_state
;
90 bool m_progressBarMarquee
;
93 // Bit field that indicates fields that have been modified by the
94 // main thread so the task dialog runner knows what to update.
98 // Runner thread that takes care of displaying and updating the
100 class wxProgressDialogTaskRunner
: public wxThread
103 wxProgressDialogTaskRunner()
104 : wxThread(wxTHREAD_JOINABLE
)
107 wxProgressDialogSharedData
* GetSharedDataObject()
108 { return &m_sharedData
; }
111 wxProgressDialogSharedData m_sharedData
;
113 virtual void* Entry();
115 static HRESULT CALLBACK
TaskDialogCallbackProc(HWND hwnd
,
122 // ============================================================================
124 // ============================================================================
129 // This function returns true if the progress dialog with the given style
130 // (combination of wxPD_XXX constants) needs the "Close" button and this button
131 // only, i.e. not a "Cancel" one.
132 bool UsesCloseButtonOnly(long style
)
134 return !(style
& (wxPD_CAN_ABORT
| wxPD_AUTO_HIDE
));
137 BOOL CALLBACK
DisplayCloseButton(HWND hwnd
, LPARAM lParam
)
139 wxProgressDialogSharedData
*sharedData
=
140 (wxProgressDialogSharedData
*) lParam
;
142 if ( wxGetWindowText( hwnd
) == sharedData
->m_labelCancel
)
144 sharedData
->m_labelCancel
= _("Close");
145 SendMessage( hwnd
, WM_SETTEXT
, 0,
146 (LPARAM
) sharedData
->m_labelCancel
.wx_str() );
154 void PerformNotificationUpdates(HWND hwnd
,
155 wxProgressDialogSharedData
*sharedData
)
157 // Update the appropriate dialog fields.
158 if ( sharedData
->m_notifications
& wxSPDD_RANGE_CHANGED
)
161 TDM_SET_PROGRESS_BAR_RANGE
,
163 MAKELPARAM(0, sharedData
->m_range
) );
166 if ( sharedData
->m_notifications
& wxSPDD_VALUE_CHANGED
)
169 TDM_SET_PROGRESS_BAR_POS
,
174 if ( sharedData
->m_notifications
& wxSPDD_PBMARQUEE_CHANGED
)
176 BOOL val
= sharedData
->m_progressBarMarquee
? TRUE
: FALSE
;
178 TDM_SET_MARQUEE_PROGRESS_BAR
,
182 TDM_SET_PROGRESS_BAR_MARQUEE
,
187 if ( sharedData
->m_notifications
& wxSPDD_TITLE_CHANGED
)
188 ::SetWindowText( hwnd
, sharedData
->m_title
.wx_str() );
190 if ( sharedData
->m_notifications
& wxSPDD_MESSAGE_CHANGED
)
192 // Split the message in the title string and the rest if it has
195 title
= sharedData
->m_message
,
198 const size_t posNL
= title
.find('\n');
199 if ( posNL
!= wxString::npos
)
201 // There can an extra new line between the first and subsequent
202 // lines to separate them as it looks better with the generic
203 // version -- but in this one, they're already separated by the use
204 // of different dialog elements, so suppress the extra new line.
206 if ( posNL
< title
.length() - 1 && title
[posNL
+ 1] == '\n' )
209 body
.assign(title
, posNL
+ numNLs
, wxString::npos
);
214 TDM_SET_ELEMENT_TEXT
,
215 TDE_MAIN_INSTRUCTION
,
216 (LPARAM
) title
.wx_str() );
219 TDM_SET_ELEMENT_TEXT
,
221 (LPARAM
) body
.wx_str() );
224 if ( sharedData
->m_notifications
& wxSPDD_EXPINFO_CHANGED
)
226 const wxString
& expandedInformation
=
227 sharedData
->m_expandedInformation
;
228 if ( !expandedInformation
.empty() )
231 TDM_SET_ELEMENT_TEXT
,
232 TDE_EXPANDED_INFORMATION
,
233 (LPARAM
) expandedInformation
.wx_str() );
237 if ( sharedData
->m_notifications
& wxSPDD_ENABLE_SKIP
)
238 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, TRUE
);
240 if ( sharedData
->m_notifications
& wxSPDD_ENABLE_ABORT
)
241 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, TRUE
);
243 if ( sharedData
->m_notifications
& wxSPDD_DISABLE_SKIP
)
244 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, FALSE
);
246 if ( sharedData
->m_notifications
& wxSPDD_DISABLE_ABORT
)
247 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, FALSE
);
249 // Is the progress finished?
250 if ( sharedData
->m_notifications
& wxSPDD_FINISHED
)
252 sharedData
->m_state
= wxGenericProgressDialog::Finished
;
254 if ( !(sharedData
->m_style
& wxPD_AUTO_HIDE
) )
256 // Change Cancel into Close and activate the button.
257 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, FALSE
);
258 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, TRUE
);
259 ::EnumChildWindows( hwnd
, DisplayCloseButton
,
260 (LPARAM
) sharedData
);
265 } // anonymous namespace
267 #endif // wxHAS_MSW_TASKDIALOG
269 // ============================================================================
270 // wxProgressDialog implementation
271 // ============================================================================
273 wxProgressDialog::wxProgressDialog( const wxString
& title
,
274 const wxString
& message
,
278 : wxGenericProgressDialog(parent
, maximum
, style
),
279 m_taskDialogRunner(NULL
),
284 #ifdef wxHAS_MSW_TASKDIALOG
285 if ( HasNativeProgressDialog() )
288 DisableOtherWindows();
292 #endif // wxHAS_MSW_TASKDIALOG
294 Create(title
, message
, maximum
, parent
, style
);
297 wxProgressDialog::~wxProgressDialog()
299 #ifdef wxHAS_MSW_TASKDIALOG
300 if ( !m_taskDialogRunner
)
305 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
306 m_sharedData
->m_notifications
|= wxSPDD_DESTROYED
;
309 m_taskDialogRunner
->Wait();
311 delete m_taskDialogRunner
;
313 ReenableOtherWindows();
315 if ( GetTopParent() )
316 GetTopParent()->Raise();
317 #endif // wxHAS_MSW_TASKDIALOG
320 bool wxProgressDialog::Update(int value
, const wxString
& newmsg
, bool *skip
)
322 #ifdef wxHAS_MSW_TASKDIALOG
323 if ( HasNativeProgressDialog() )
325 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
327 // Do nothing in canceled state.
328 if ( !DoNativeBeforeUpdate(skip
) )
333 wxASSERT_MSG( value
<= m_maximum
, wxT("invalid progress value") );
335 m_sharedData
->m_value
= value
;
336 m_sharedData
->m_notifications
|= wxSPDD_VALUE_CHANGED
;
338 if ( !newmsg
.empty() )
341 m_sharedData
->m_message
= newmsg
;
342 m_sharedData
->m_notifications
|= wxSPDD_MESSAGE_CHANGED
;
345 if ( m_sharedData
->m_progressBarMarquee
)
347 m_sharedData
->m_progressBarMarquee
= false;
348 m_sharedData
->m_notifications
|= wxSPDD_PBMARQUEE_CHANGED
;
351 UpdateExpandedInformation( value
);
353 // Has the progress bar finished?
354 if ( value
== m_maximum
)
356 if ( m_state
== Finished
)
360 m_sharedData
->m_state
= Finished
;
361 m_sharedData
->m_notifications
|= wxSPDD_FINISHED
;
362 if( !HasPDFlag(wxPD_AUTO_HIDE
) && newmsg
.empty() )
364 // Provide the finishing message if the application didn't.
365 m_message
= _("Done.");
366 m_sharedData
->m_message
= m_message
;
367 m_sharedData
->m_notifications
|= wxSPDD_MESSAGE_CHANGED
;
371 return m_sharedData
->m_state
!= Canceled
;
373 #endif // wxHAS_MSW_TASKDIALOG
375 return wxGenericProgressDialog::Update( value
, newmsg
, skip
);
378 bool wxProgressDialog::Pulse(const wxString
& newmsg
, bool *skip
)
380 #ifdef wxHAS_MSW_TASKDIALOG
381 if ( HasNativeProgressDialog() )
383 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
385 // Do nothing in canceled state.
386 if ( !DoNativeBeforeUpdate(skip
) )
389 if ( !m_sharedData
->m_progressBarMarquee
)
391 m_sharedData
->m_progressBarMarquee
= true;
392 m_sharedData
->m_notifications
|= wxSPDD_PBMARQUEE_CHANGED
;
395 if ( !newmsg
.empty() )
398 m_sharedData
->m_message
= newmsg
;
399 m_sharedData
->m_notifications
|= wxSPDD_MESSAGE_CHANGED
;
402 // The value passed here doesn't matter, only elapsed time makes sense
403 // in indeterminate mode anyhow.
404 UpdateExpandedInformation(0);
406 return m_sharedData
->m_state
!= Canceled
;
408 #endif // wxHAS_MSW_TASKDIALOG
410 return wxGenericProgressDialog::Pulse( newmsg
, skip
);
413 bool wxProgressDialog::DoNativeBeforeUpdate(bool *skip
)
415 #ifdef wxHAS_MSW_TASKDIALOG
416 if ( HasNativeProgressDialog() )
418 if ( m_sharedData
->m_skipped
)
420 if ( skip
&& !*skip
)
423 m_sharedData
->m_skipped
= false;
424 m_sharedData
->m_notifications
|= wxSPDD_ENABLE_SKIP
;
428 if ( m_sharedData
->m_state
== Canceled
)
429 m_timeStop
= m_sharedData
->m_timeStop
;
431 return m_sharedData
->m_state
!= Canceled
;
433 #endif // wxHAS_MSW_TASKDIALOG
436 wxFAIL_MSG( "unreachable" );
441 void wxProgressDialog::Resume()
443 wxGenericProgressDialog::Resume();
445 #ifdef wxHAS_MSW_TASKDIALOG
446 if ( HasNativeProgressDialog() )
451 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
452 m_sharedData
->m_state
= m_state
;
454 // "Skip" was disabled when "Cancel" had been clicked, so re-enable
456 m_sharedData
->m_notifications
|= wxSPDD_ENABLE_SKIP
;
458 if ( !UsesCloseButtonOnly(GetPDStyle()) )
459 m_sharedData
->m_notifications
|= wxSPDD_ENABLE_ABORT
;
461 hwnd
= m_sharedData
->m_hwnd
;
462 } // Unlock m_cs, we can't call any function operating on a dialog with
463 // it locked as it can result in a deadlock if the dialog callback is
464 // called by Windows.
466 // After resuming we need to bring the window on top of the Z-order as
467 // it could be hidden by another window shown from the main thread,
468 // e.g. a confirmation dialog asking whether the user really wants to
471 // Notice that this must be done from the main thread as it owns the
472 // currently active window and attempts to do this from the task dialog
473 // thread would simply fail.
474 ::BringWindowToTop(hwnd
);
476 #endif // wxHAS_MSW_TASKDIALOG
479 int wxProgressDialog::GetValue() const
481 #ifdef wxHAS_MSW_TASKDIALOG
482 if ( HasNativeProgressDialog() )
484 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
485 return m_sharedData
->m_value
;
487 #endif // wxHAS_MSW_TASKDIALOG
489 return wxGenericProgressDialog::GetValue();
492 wxString
wxProgressDialog::GetMessage() const
494 #ifdef wxHAS_MSW_TASKDIALOG
495 if ( HasNativeProgressDialog() )
497 #endif // wxHAS_MSW_TASKDIALOG
499 return wxGenericProgressDialog::GetMessage();
502 void wxProgressDialog::SetRange(int maximum
)
504 wxGenericProgressDialog::SetRange( maximum
);
506 #ifdef wxHAS_MSW_TASKDIALOG
507 if ( HasNativeProgressDialog() )
509 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
511 m_sharedData
->m_range
= maximum
;
512 m_sharedData
->m_notifications
|= wxSPDD_RANGE_CHANGED
;
514 #endif // wxHAS_MSW_TASKDIALOG
517 bool wxProgressDialog::WasSkipped() const
519 #ifdef wxHAS_MSW_TASKDIALOG
520 if ( HasNativeProgressDialog() )
524 // Couldn't be skipped before being shown.
528 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
529 return m_sharedData
->m_skipped
;
531 #endif // wxHAS_MSW_TASKDIALOG
533 return wxGenericProgressDialog::WasSkipped();
536 bool wxProgressDialog::WasCancelled() const
538 #ifdef wxHAS_MSW_TASKDIALOG
539 if ( HasNativeProgressDialog() )
541 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
542 return m_sharedData
->m_state
== Canceled
;
544 #endif // wxHAS_MSW_TASKDIALOG
546 return wxGenericProgressDialog::WasCancelled();
549 void wxProgressDialog::SetTitle(const wxString
& title
)
551 #ifdef wxHAS_MSW_TASKDIALOG
552 if ( HasNativeProgressDialog() )
558 wxCriticalSectionLocker
locker(m_sharedData
->m_cs
);
559 m_sharedData
->m_title
= title
;
560 m_sharedData
->m_notifications
= wxSPDD_TITLE_CHANGED
;
563 #endif // wxHAS_MSW_TASKDIALOG
565 wxGenericProgressDialog::SetTitle(title
);
568 wxString
wxProgressDialog::GetTitle() const
570 #ifdef wxHAS_MSW_TASKDIALOG
571 if ( HasNativeProgressDialog() )
573 #endif // wxHAS_MSW_TASKDIALOG
575 return wxGenericProgressDialog::GetTitle();
578 bool wxProgressDialog::Show(bool show
)
580 #ifdef wxHAS_MSW_TASKDIALOG
581 if ( HasNativeProgressDialog() )
583 // The dialog can't be hidden at all and showing it again after it had
584 // been shown before doesn't do anything.
585 if ( !show
|| m_taskDialogRunner
)
588 // We're showing the dialog for the first time, create the thread that
590 m_taskDialogRunner
= new wxProgressDialogTaskRunner
;
591 m_sharedData
= m_taskDialogRunner
->GetSharedDataObject();
593 // Initialize shared data.
594 m_sharedData
->m_title
= m_title
;
595 m_sharedData
->m_message
= m_message
;
596 m_sharedData
->m_range
= m_maximum
;
597 m_sharedData
->m_state
= Uncancelable
;
598 m_sharedData
->m_style
= GetPDStyle();
600 if ( HasPDFlag(wxPD_CAN_ABORT
) )
602 m_sharedData
->m_state
= Continue
;
603 m_sharedData
->m_labelCancel
= _("Cancel");
605 else if ( !HasPDFlag(wxPD_AUTO_HIDE
) )
607 m_sharedData
->m_labelCancel
= _("Close");
610 if ( HasPDFlag(wxPD_ELAPSED_TIME
|
611 wxPD_ESTIMATED_TIME
|
612 wxPD_REMAINING_TIME
) )
614 // Use a non-empty string just to have the collapsible pane shown.
615 m_sharedData
->m_expandedInformation
= " ";
618 // Do launch the thread.
619 if ( m_taskDialogRunner
->Create() != wxTHREAD_NO_ERROR
)
621 wxLogError( "Unable to create thread!" );
625 if ( m_taskDialogRunner
->Run() != wxTHREAD_NO_ERROR
)
627 wxLogError( "Unable to start thread!" );
631 // Do not show the underlying dialog.
634 #endif // wxHAS_MSW_TASKDIALOG
636 return wxGenericProgressDialog::Show( show
);
639 bool wxProgressDialog::HasNativeProgressDialog() const
641 #ifdef wxHAS_MSW_TASKDIALOG
642 // Native task dialog, if available, can't be used without any buttons so
643 // we fall back to the generic one if none of "Skip", "Cancel" and "Close"
645 return HasNativeTaskDialog()
646 && (HasPDFlag(wxPD_CAN_SKIP
| wxPD_CAN_ABORT
) ||
647 !HasPDFlag(wxPD_AUTO_HIDE
));
648 #else // !wxHAS_MSW_TASKDIALOG
649 // This shouldn't be even called in !wxHAS_MSW_TASKDIALOG case but as we
650 // still must define the function as returning something, return false.
652 #endif // wxHAS_MSW_TASKDIALOG/!wxHAS_MSW_TASKDIALOG
655 void wxProgressDialog::UpdateExpandedInformation(int value
)
657 #ifdef wxHAS_MSW_TASKDIALOG
658 unsigned long elapsedTime
;
659 unsigned long estimatedTime
;
660 unsigned long remainingTime
;
661 UpdateTimeEstimates(value
, elapsedTime
, estimatedTime
, remainingTime
);
663 int realEstimatedTime
= estimatedTime
,
664 realRemainingTime
= remainingTime
;
665 if ( m_sharedData
->m_progressBarMarquee
)
667 // In indeterminate mode we don't have any estimation neither for the
668 // remaining nor for estimated time.
670 realRemainingTime
= -1;
673 wxString expandedInformation
;
675 // Calculate the three different timing values.
676 if ( HasPDFlag(wxPD_ELAPSED_TIME
) )
678 expandedInformation
<< GetElapsedLabel()
680 << GetFormattedTime(elapsedTime
);
683 if ( HasPDFlag(wxPD_ESTIMATED_TIME
) )
685 if ( !expandedInformation
.empty() )
686 expandedInformation
+= "\n";
688 expandedInformation
<< GetEstimatedLabel()
690 << GetFormattedTime(realEstimatedTime
);
693 if ( HasPDFlag(wxPD_REMAINING_TIME
) )
695 if ( !expandedInformation
.empty() )
696 expandedInformation
+= "\n";
698 expandedInformation
<< GetRemainingLabel()
700 << GetFormattedTime(realRemainingTime
);
703 // Update with new timing information.
704 if ( expandedInformation
!= m_sharedData
->m_expandedInformation
)
706 m_sharedData
->m_expandedInformation
= expandedInformation
;
707 m_sharedData
->m_notifications
|= wxSPDD_EXPINFO_CHANGED
;
709 #else // !wxHAS_MSW_TASKDIALOG
711 #endif // wxHAS_MSW_TASKDIALOG/!wxHAS_MSW_TASKDIALOG
714 // ----------------------------------------------------------------------------
715 // wxProgressDialogTaskRunner and related methods
716 // ----------------------------------------------------------------------------
718 #ifdef wxHAS_MSW_TASKDIALOG
720 void* wxProgressDialogTaskRunner::Entry()
722 WinStruct
<TASKDIALOGCONFIG
> tdc
;
723 wxMSWTaskDialogConfig wxTdc
;
726 wxCriticalSectionLocker
locker(m_sharedData
.m_cs
);
728 wxTdc
.caption
= m_sharedData
.m_title
.wx_str();
729 wxTdc
.message
= m_sharedData
.m_message
.wx_str();
731 wxTdc
.MSWCommonTaskDialogInit( tdc
);
732 tdc
.pfCallback
= TaskDialogCallbackProc
;
733 tdc
.lpCallbackData
= (LONG_PTR
) &m_sharedData
;
735 // Undo some of the effects of MSWCommonTaskDialogInit().
736 tdc
.dwFlags
&= ~TDF_EXPAND_FOOTER_AREA
; // Expand in content area.
737 tdc
.dwCommonButtons
= 0; // Don't use common buttons.
739 wxTdc
.useCustomLabels
= true;
741 if ( m_sharedData
.m_style
& wxPD_CAN_SKIP
)
742 wxTdc
.AddTaskDialogButton( tdc
, Id_SkipBtn
, 0, _("Skip") );
744 // Use a Cancel button when requested or use a Close button when
745 // the dialog does not automatically hide.
746 if ( (m_sharedData
.m_style
& wxPD_CAN_ABORT
)
747 || !(m_sharedData
.m_style
& wxPD_AUTO_HIDE
) )
749 wxTdc
.AddTaskDialogButton( tdc
, IDCANCEL
, 0,
750 m_sharedData
.m_labelCancel
);
753 tdc
.dwFlags
|= TDF_CALLBACK_TIMER
| TDF_SHOW_PROGRESS_BAR
;
755 if ( !m_sharedData
.m_expandedInformation
.empty() )
757 tdc
.pszExpandedInformation
=
758 m_sharedData
.m_expandedInformation
.wx_str();
762 TaskDialogIndirect_t taskDialogIndirect
= GetTaskDialogIndirectFunc();
763 if ( !taskDialogIndirect
)
767 HRESULT hr
= taskDialogIndirect(&tdc
, &msAns
, NULL
, NULL
);
769 wxLogApiError( "TaskDialogIndirect", hr
);
776 wxProgressDialogTaskRunner::TaskDialogCallbackProc
781 LPARAM
WXUNUSED(lParam
),
785 wxProgressDialogSharedData
* const sharedData
=
786 (wxProgressDialogSharedData
*) dwRefData
;
788 wxCriticalSectionLocker
locker(sharedData
->m_cs
);
790 switch ( uNotification
)
793 // Store the HWND for the main thread use.
794 sharedData
->m_hwnd
= hwnd
;
796 // Set the maximum value and disable Close button.
798 TDM_SET_PROGRESS_BAR_RANGE
,
800 MAKELPARAM(0, sharedData
->m_range
) );
802 if ( UsesCloseButtonOnly(sharedData
->m_style
) )
803 ::SendMessage( hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, FALSE
);
806 case TDN_BUTTON_CLICKED
:
810 ::SendMessage(hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, FALSE
);
811 sharedData
->m_skipped
= true;
815 if ( sharedData
->m_state
816 == wxGenericProgressDialog::Finished
)
821 // Close button on the window triggers an IDCANCEL press,
822 // don't allow it when it should only be possible to close
823 // a finished dialog.
824 if ( !UsesCloseButtonOnly(sharedData
->m_style
) )
826 wxCHECK_MSG( sharedData
->m_state
==
827 wxGenericProgressDialog::Continue
,
829 "Dialog not in a cancelable state!");
831 ::SendMessage(hwnd
, TDM_ENABLE_BUTTON
, Id_SkipBtn
, FALSE
);
832 ::SendMessage(hwnd
, TDM_ENABLE_BUTTON
, IDCANCEL
, FALSE
);
834 sharedData
->m_timeStop
= wxGetCurrentTime();
835 sharedData
->m_state
= wxGenericProgressDialog::Canceled
;
843 PerformNotificationUpdates(hwnd
, sharedData
);
845 // End dialog in three different cases:
846 // 1. Progress finished and dialog should automatically hide.
847 // 2. The wxProgressDialog object was destructed and should
848 // automatically hide.
849 // 3. The dialog was canceled and wxProgressDialog object
852 sharedData
->m_state
== wxGenericProgressDialog::Canceled
;
854 sharedData
->m_state
== wxGenericProgressDialog::Finished
;
856 (sharedData
->m_notifications
& wxSPDD_DESTROYED
) != 0;
857 bool shouldAutoHide
= (sharedData
->m_style
& wxPD_AUTO_HIDE
) != 0;
859 if ( (shouldAutoHide
&& (isFinished
|| wasDestroyed
))
860 || (wasDestroyed
&& isCanceled
) )
862 ::EndDialog( hwnd
, IDCLOSE
);
865 sharedData
->m_notifications
= 0;
874 #endif // wxHAS_MSW_TASKDIALOG
876 #endif // wxUSE_PROGRESSDLG && wxUSE_THREADS