1 // TortoiseGit - a Windows shell extension for easy version control
3 // Copyright (C) 2010-2012 - TortoiseGit
4 // Copyright (C) 2003-2006,2008-2011 - TortoiseSVN
6 // This program is free software; you can redistribute it and/or
7 // modify it under the terms of the GNU General Public License
8 // as published by the Free Software Foundation; either version 2
9 // of the License, or (at your option) any later version.
11 // This program is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 // GNU General Public License for more details.
16 // You should have received a copy of the GNU General Public License
17 // along with this program; if not, write to the Free Software Foundation,
18 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21 #include "SysProgressDlg.h"
23 CSysProgressDlg::CSysProgressDlg()
26 , m_dwDlgFlags(PROGDLG_NORMAL
)
33 CSysProgressDlg::~CSysProgressDlg()
37 if (m_isVisible
) //still visible, so stop first before destroying
38 m_pIDlg
->StopProgressDialog();
45 bool CSysProgressDlg::EnsureValid()
50 HRESULT hr
= m_pIDlg
.CoCreateInstance (CLSID_ProgressDialog
, NULL
, CLSCTX_INPROC_SERVER
);
51 return (SUCCEEDED(hr
));
54 void CSysProgressDlg::SetTitle(LPCTSTR szTitle
)
59 m_pIDlg
->SetTitle(T2COLE(szTitle
));
62 void CSysProgressDlg::SetTitle ( UINT idTitle
)
64 SetTitle(CString(MAKEINTRESOURCE(idTitle
)));
67 void CSysProgressDlg::SetLine(DWORD dwLine
, LPCTSTR szText
, bool bCompactPath
/* = false */)
72 m_pIDlg
->SetLine(dwLine
, T2COLE(szText
), bCompactPath
, NULL
);
77 void CSysProgressDlg::SetCancelMsg ( UINT idMessage
)
79 SetCancelMsg(CString(MAKEINTRESOURCE(idMessage
)));
83 void CSysProgressDlg::SetCancelMsg(LPCTSTR szMessage
)
88 m_pIDlg
->SetCancelMsg(T2COLE(szMessage
), NULL
);
92 void CSysProgressDlg::SetAnimation(HINSTANCE hinst
, UINT uRsrcID
)
96 m_pIDlg
->SetAnimation(hinst
, uRsrcID
);
100 void CSysProgressDlg::SetAnimation(UINT uRsrcID
)
104 m_pIDlg
->SetAnimation(AfxGetResourceHandle(), uRsrcID
);
108 void CSysProgressDlg::SetTime(bool bTime
/* = true */)
110 m_dwDlgFlags
&= ~(PROGDLG_NOTIME
| PROGDLG_AUTOTIME
);
113 m_dwDlgFlags
|= PROGDLG_AUTOTIME
;
115 m_dwDlgFlags
|= PROGDLG_NOTIME
;
118 void CSysProgressDlg::SetShowProgressBar(bool bShow
/* = true */)
121 m_dwDlgFlags
&= ~PROGDLG_NOPROGRESSBAR
;
123 m_dwDlgFlags
|= PROGDLG_NOPROGRESSBAR
;
126 HRESULT
CSysProgressDlg::ShowModal (CWnd
* pwndParent
, BOOL immediately
/* = true */)
129 return ShowModal(pwndParent
->GetSafeHwnd(), immediately
);
132 HRESULT
CSysProgressDlg::ShowModeless(CWnd
* pwndParent
, BOOL immediately
)
135 return ShowModeless(pwndParent
->GetSafeHwnd(), immediately
);
138 void CSysProgressDlg::FormatPathLine ( DWORD dwLine
, UINT idFormatText
, ...)
141 va_start(args
, idFormatText
);
144 sText
.FormatV(CString(MAKEINTRESOURCE(idFormatText
)), args
);
145 SetLine(dwLine
, sText
, true);
150 void CSysProgressDlg::FormatPathLine ( DWORD dwLine
, CString FormatText
, ...)
153 va_start(args
, FormatText
);
156 sText
.FormatV(FormatText
, args
);
157 SetLine(dwLine
, sText
, true);
162 void CSysProgressDlg::FormatNonPathLine(DWORD dwLine
, UINT idFormatText
, ...)
165 va_start(args
, idFormatText
);
168 sText
.FormatV(CString(MAKEINTRESOURCE(idFormatText
)), args
);
169 SetLine(dwLine
, sText
, false);
174 void CSysProgressDlg::FormatNonPathLine(DWORD dwLine
, CString FormatText
, ...)
177 va_start(args
, FormatText
);
180 sText
.FormatV(FormatText
, args
);
181 SetLine(dwLine
, sText
, false);
187 HRESULT
CSysProgressDlg::ShowModal(HWND hWndParent
, BOOL immediately
/* = true */)
190 m_hWndProgDlg
= NULL
;
193 m_hWndParent
= hWndParent
;
194 HRESULT hr
= m_pIDlg
->StartProgressDialog(hWndParent
, NULL
, m_dwDlgFlags
| PROGDLG_MODAL
, NULL
);
198 ATL::CComPtr
<IOleWindow
> pOleWindow
;
199 HRESULT hr2
= m_pIDlg
.QueryInterface(&pOleWindow
);
202 hr2
= pOleWindow
->GetWindow(&m_hWndProgDlg
);
205 // StartProgressDialog creates a new thread to host the progress window.
206 // When the window receives WM_DESTROY message StopProgressDialog() wrongly
207 // attempts to re-enable the parent in the calling thread (our thread),
208 // after the progress window is destroyed and the progress thread has died.
209 // When the progress window dies, the system tries to assign a new foreground window.
210 // It cannot assign to hwndParent because StartProgressDialog (w/PROGDLG_MODAL) disabled the parent window.
211 // So the system hands the foreground activation to the next process that wants it in the
212 // system foreground queue. Thus we lose our right to recapture the foreground window.
213 // The way to fix this bug is to insert a call to EnableWindow(hWndParent) in the WM_DESTROY
214 // handler for the progress window in the progress thread.
216 // To do that, we Subclass the progress dialog
217 // Since the window and thread created by the progress dialog object live on a few
218 // milliseconds after calling Stop() and Release(), we must not store anything
219 // in member variables of this class but must only store everything in the window
220 // itself: thus we use SetProp()/GetProp() to store the data.
223 m_OrigProc
= (WNDPROC
) SetWindowLongPtr(m_hWndProgDlg
, GWLP_WNDPROC
, (LONG_PTR
) fnSubclass
);
224 SetProp(m_hWndProgDlg
, L
"ParentWindow", m_hWndParent
);
225 SetProp(m_hWndProgDlg
, L
"OrigProc", m_OrigProc
);
228 ShowWindow(m_hWndProgDlg
, SW_SHOW
);
236 HRESULT
CSysProgressDlg::ShowModeless(HWND hWndParent
, BOOL immediately
)
239 m_hWndProgDlg
= NULL
;
242 m_hWndParent
= hWndParent
;
243 HRESULT hr
= m_pIDlg
->StartProgressDialog(hWndParent
, NULL
, m_dwDlgFlags
, NULL
);
247 ATL::CComPtr
<IOleWindow
> pOleWindow
;
248 HRESULT hr2
= m_pIDlg
.QueryInterface(&pOleWindow
);
251 hr2
= pOleWindow
->GetWindow(&m_hWndProgDlg
);
254 // see comment in ShowModal() for why we subclass the window
257 m_OrigProc
= (WNDPROC
) SetWindowLongPtr(m_hWndProgDlg
, GWLP_WNDPROC
, (LONG_PTR
) fnSubclass
);
258 SetProp(m_hWndProgDlg
, L
"ParentWindow", m_hWndParent
);
259 SetProp(m_hWndProgDlg
, L
"OrigProc", m_OrigProc
);
262 ShowWindow(m_hWndProgDlg
, SW_SHOW
);
269 void CSysProgressDlg::SetProgress(DWORD dwProgress
, DWORD dwMax
)
273 m_pIDlg
->SetProgress(dwProgress
, dwMax
);
277 void CSysProgressDlg::SetProgress64(ULONGLONG u64Progress
, ULONGLONG u64ProgressMax
)
281 m_pIDlg
->SetProgress64(u64Progress
, u64ProgressMax
);
285 bool CSysProgressDlg::HasUserCancelled()
290 return (0 != m_pIDlg
->HasUserCancelled());
293 void CSysProgressDlg::Stop()
295 if ((m_isVisible
)&&(IsValid()))
297 m_pIDlg
->StopProgressDialog();
298 // Sometimes the progress dialog sticks around after stopping it,
299 // until the mouse pointer is moved over it or some other triggers.
300 // We hide the window here immediately.
303 ShowWindow(m_hWndProgDlg
, SW_HIDE
);
308 m_hWndProgDlg
= NULL
;
312 void CSysProgressDlg::ResetTimer()
316 m_pIDlg
->Timer(PDTIMER_RESET
, NULL
);
320 LRESULT
CSysProgressDlg::fnSubclass(HWND hwnd
,UINT uMsg
,WPARAM wParam
,LPARAM lParam
)
322 LONG_PTR origproc
= (LONG_PTR
)GetProp(hwnd
, L
"OrigProc");
323 if (uMsg
== WM_DESTROY
)
325 HWND hParent
= (HWND
)GetProp(hwnd
, L
"ParentWindow");
326 EnableWindow(hParent
, TRUE
);
328 SetWindowLongPtr (hwnd
, GWLP_WNDPROC
, origproc
);
329 RemoveProp(hwnd
, L
"ParentWindow");
330 RemoveProp(hwnd
, L
"OrigProc");
332 return CallWindowProc ((WNDPROC
)origproc
, hwnd
, uMsg
, wParam
, lParam
);