1 // TortoiseGit - a Windows shell extension for easy version control
3 // Copyright (C) 2008-2013 - TortoiseGit
5 // This program is free software; you can redistribute it and/or
6 // modify it under the terms of the GNU General Public License
7 // as published by the Free Software Foundation; either version 2
8 // of the License, or (at your option) any later version.
10 // This program is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 // GNU General Public License for more details.
15 // You should have received a copy of the GNU General Public License
16 // along with this program; if not, write to the Free Software Foundation,
17 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 // GitLogList.cpp : implementation file
23 #include "HintListCtrl.h"
25 #include "ProjectProperties.h"
28 #include "SplitterControl.h"
30 #include "MenuButton.h"
31 #include "LogDlgHelper.h"
32 #include "FilterEdit.h"
35 //#include "GitLogList.h"
37 #include "GitLogCache.h"
39 #include <GitStatusListCtrl.h>
43 #if (NTDDI_VERSION < NTDDI_LONGHORN)
45 enum LISTITEMSTATES_MINE
{
50 LISS_SELECTEDNOTFOCUS
= 5,
54 // these defines must be in the order the columns are inserted!
57 #define MCS_NOTRAILINGDATES 0x0040
58 #define MCS_SHORTDAYSOFWEEK 0x0080
59 #define MCS_NOSELCHANGEONNAV 0x0100
61 #define DTM_SETMCSTYLE (DTM_FIRST + 11)
65 #define ICONITEMBORDER 5
67 #define GITLOG_START 0
68 #define GITLOG_START_ALL 1
69 #define GITLOG_END 100
71 #define LOGFILTER_ALL 0xFFFF
72 #define LOGFILTER_MESSAGES 0x0001
73 #define LOGFILTER_PATHS 0x0002
74 #define LOGFILTER_AUTHORS 0x0004
75 #define LOGFILTER_REVS 0x0008
76 #define LOGFILTER_REGEX 0x0010
77 #define LOGFILTER_BUGID 0x0020
78 #define LOGFILTER_SUBJECT 0x0040
79 #define LOGFILTER_REFNAME 0x0080
80 #define LOGFILTER_EMAILS 0x0100
82 #define LOGLIST_SHOWNOREFS 0x0000
83 #define LOGLIST_SHOWLOCALBRANCHES 0x0001
84 #define LOGLIST_SHOWREMOTEBRANCHES 0x0002
85 #define LOGLIST_SHOWTAGS 0x0004
86 #define LOGLIST_SHOWSTASH 0x0008
87 #define LOGLIST_SHOWBISECT 0x0010
88 #define LOGLIST_SHOWALLREFS 0xFFFF
90 //typedef void CALLBACK_PROCESS(void * data, int progress);
91 #define MSG_LOADED (WM_USER+110)
92 #define MSG_LOAD_PERCENTAGE (WM_USER+111)
93 #define MSG_REFLOG_CHANGED (WM_USER+112)
94 #define MSG_FETCHED_DIFF (WM_USER+113)
96 class CThreadSafePtrArray
: public CPtrArray
98 CComCriticalSection
*m_critSec
;
100 CThreadSafePtrArray(CComCriticalSection
*section
){ m_critSec
= section
;}
101 void * SafeGetAt(INT_PTR i
)
106 if( i
<0 || i
>=GetCount())
119 INT_PTR
SafeAdd(void *newElement
)
124 ret
= Add(newElement
);
141 class CGitLogListBase
: public CHintListCtrl
143 DECLARE_DYNAMIC(CGitLogListBase
)
147 virtual ~CGitLogListBase();
148 ProjectProperties m_ProjectProperties
;
150 CFilterData m_Filter
;
152 void UpdateProjectProperties()
154 // do not crash if config file is broken; needed for TortoiseGitBlame
157 m_ProjectProperties
.ReadProps(this->m_Path
);
163 if ((!m_ProjectProperties
.sUrl
.IsEmpty())||(!m_ProjectProperties
.sCheckRe
.IsEmpty()))
164 m_bShowBugtraqColumn
= true;
166 m_bShowBugtraqColumn
= false;
169 void ResetWcRev(bool refresh
= false)
172 m_wcRev
.GetSubject() = CString(MAKEINTRESOURCE(IDS_LOG_WORKINGDIRCHANGES
));
173 m_wcRev
.m_Mark
= _T('-');
174 m_wcRev
.GetBody() = CString(MAKEINTRESOURCE(IDS_LOG_FETCHINGSTATUS
));
175 m_wcRev
.m_CallDiffAsync
= DiffAsync
;
176 InterlockedExchange(&m_wcRev
.m_IsDiffFiles
, FALSE
);
177 if (refresh
&& m_bShowWC
)
178 m_arShownList
[0] = &m_wcRev
;
180 void SetProjectPropertiesPath(const CTGitPath
& path
) {m_ProjectProperties
.ReadProps(path
);}
182 volatile LONG m_bNoDispUpdates
;
183 BOOL m_IsIDReplaceAction
;
185 void hideFromContextMenu(unsigned __int64 hideMask
, bool exclusivelyShow
);
186 BOOL m_IsRebaseReplaceGraph
;
187 BOOL m_bNoHightlightHead
;
189 void MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct
);
191 BOOL m_bStrictStopped
;
192 BOOL m_bShowBugtraqColumn
;
195 unsigned __int64 m_ContextMenuMask
;
200 volatile LONG m_bThreadRunning
;
201 CLogCache m_LogCache
;
204 // don't forget to bump BLAME_COLUMN_VERSION in GitStatusListCtrlHelpers.cpp if you change columns
217 LOGLIST_COMMIT_EMAIL
,
220 LOGLIST_MESSAGE_MAX
=300,
221 LOGLIST_MESSAGE_MIN
=200,
223 GIT_LOG_GRAPH
= 1<< LOGLIST_GRAPH
,
224 GIT_LOG_REBASE
= 1<< LOGLIST_REBASE
,
225 GIT_LOG_ID
= 1<< LOGLIST_ID
,
226 GIT_LOG_HASH
= 1<< LOGLIST_HASH
,
227 GIT_LOG_ACTIONS
= 1<< LOGLIST_ACTION
,
228 GIT_LOG_MESSAGE
= 1<< LOGLIST_MESSAGE
,
229 GIT_LOG_AUTHOR
= 1<< LOGLIST_AUTHOR
,
230 GIT_LOG_DATE
= 1<< LOGLIST_DATE
,
231 GIT_LOG_EMAIL
= 1<< LOGLIST_EMAIL
,
232 GIT_LOG_COMMIT_NAME
= 1<< LOGLIST_COMMIT_NAME
,
233 GIT_LOG_COMMIT_EMAIL
= 1<< LOGLIST_COMMIT_EMAIL
,
234 GIT_LOG_COMMIT_DATE
= 1<< LOGLIST_COMMIT_DATE
,
235 GIT_LOGLIST_BUG
= 1<< LOGLIST_BUG
,
240 // needs to start with 1, since 0 is the return value if *nothing* is clicked on in the context menu
241 ID_COMPARE
= 1, // compare revision with WC
243 ID_COMPARETWO
, // compare two revisions
247 ID_GNUDIFF1
, // compare with WC, unified
248 ID_GNUDIFF2
, // compare two revisions, unified
265 ID_COMPAREWITHPREVIOUS
,
266 ID_BLAMEWITHPREVIOUS
,
281 ID_REFLOG_STASH_APPLY
,
283 ID_REBASE_TO_VERSION
,
290 ID_COPYCLIPBOARDMESSAGES
,
300 inline unsigned __int64
GetContextMenuBit(int i
){ return ((unsigned __int64
)0x1)<<i
;}
301 void InsertGitColumn();
302 void ResizeAllListCtrlCols();
303 void CopySelectionToClipBoard(int toCopy
= ID_COPY_ALL
);
304 void DiffSelectedRevWithPrevious();
305 bool IsSelectionContinuous();
307 int FillGitLog(CTGitPath
*path
, CString
*range
= NULL
, int infomask
= CGit::LOG_INFO_STAT
| CGit::LOG_INFO_FILESTATE
| CGit::LOG_INFO_SHOW_MERGEDFILE
);
308 BOOL
IsMatchFilter(bool bRegex
, GitRev
*pRev
, tr1::wregex
&pat
);
310 CFindDlg
*m_pFindDialog
;
311 static const UINT m_FindDialogMessage
;
314 static const UINT m_ScrollToMessage
;
316 inline int ShownCountWithStopped() const { return (int)m_arShownList
.GetCount() + (m_bStrictStopped
? 1 : 0); }
317 int FetchLogAsync(void * data
=NULL
);
318 CThreadSafePtrArray m_arShownList
;
319 void Refresh(BOOL IsCleanFilter
=TRUE
);
320 void RecalculateShownList(CThreadSafePtrArray
* pShownlist
);
323 DWORD m_SelectedFilters
;
324 bool m_bFilterWithRegex
;
325 CLogDataVector m_logEntries
;
328 bool ValidateRegexp(LPCTSTR regexp_str
, tr1::wregex
& pat
, bool bMatchCase
= false );
329 CString m_sFilterText
;
336 CGitHash m_lastSelectedHash
;
339 void GetTimeRange(CTime
&oldest
,CTime
&latest
);
340 virtual void ContextMenuAction(int cmd
,int FirstSelect
, int LastSelect
, CMenu
* menu
)=0;
345 if (g_Git
.GetMapHashToFriendName(m_HashMap
))
346 MessageBox(g_Git
.GetGitLastErr(_T("Could not get all refs.")), _T("TortoiseGit"), MB_ICONERROR
);
348 m_CurrentBranch
=g_Git
.GetCurrentBranch();
350 if (g_Git
.GetHash(m_HeadHash
, _T("HEAD")))
352 MessageBox(g_Git
.GetGitLastErr(_T("Could not get HEAD hash. Quitting...")), _T("TortoiseGit"), MB_ICONERROR
);
356 m_wcRev
.m_ParentHash
.clear();
357 m_wcRev
.m_ParentHash
.push_back(m_HeadHash
);
360 FetchTrackingBranchList();
362 void SafeTerminateThread()
364 if (m_LoadingThread
!=NULL
&& InterlockedExchange(&m_bExitThread
, TRUE
) == FALSE
)
366 DWORD ret
= WAIT_TIMEOUT
;
367 for (int i
= 0; i
< 200 && m_bThreadRunning
; ++i
)
368 ret
=::WaitForSingleObject(m_LoadingThread
->m_hThread
, 100);
369 if (ret
== WAIT_TIMEOUT
&& m_bThreadRunning
)
370 ::TerminateThread(m_LoadingThread
, 0);
371 m_LoadingThread
= NULL
;
375 bool IsInWorkingThread()
377 return (AfxGetThread() == m_LoadingThread
);
380 void SetRange(const CString
& range
)
385 CString
GetRange() const { return m_sRange
; }
389 volatile LONG m_bExitThread
;
390 CWinThread
* m_LoadingThread
;
391 MAP_HASH_NAME m_HashMap
;
392 std::map
<CString
, std::pair
<CString
, CString
>> m_TrackingMap
;
395 CString m_ColumnRegKey
;
401 CString simplifiedName
;
407 DECLARE_MESSAGE_MAP()
408 afx_msg
void OnDestroy();
409 virtual afx_msg
void OnNMCustomdrawLoglist(NMHDR
*pNMHDR
, LRESULT
*pResult
);
410 virtual afx_msg
void OnLvnGetdispinfoLoglist(NMHDR
*pNMHDR
, LRESULT
*pResult
);
411 afx_msg LRESULT
OnFindDialogMessage(WPARAM wParam
, LPARAM lParam
);
412 afx_msg LRESULT
OnScrollToMessage(WPARAM wParam
, LPARAM lParam
);
413 afx_msg
int OnCreate(LPCREATESTRUCT lpCreateStruct
);
414 afx_msg
void OnContextMenu(CWnd
* pWnd
, CPoint point
);
415 afx_msg LRESULT
OnLoad(WPARAM wParam
, LPARAM lParam
);
416 afx_msg
void OnHdnBegintrack(NMHDR
*pNMHDR
, LRESULT
*pResult
);
417 afx_msg
void OnHdnItemchanging(NMHDR
*pNMHDR
, LRESULT
*pResult
);
418 afx_msg
void OnColumnResized(NMHDR
*pNMHDR
, LRESULT
*pResult
);
419 afx_msg
void OnColumnMoved(NMHDR
*pNMHDR
, LRESULT
*pResult
);
420 void OnNMDblclkLoglist(NMHDR
* /*pNMHDR*/, LRESULT
*pResult
);
421 afx_msg
void OnLvnOdfinditemLoglist(NMHDR
*pNMHDR
, LRESULT
*pResult
);
422 void PreSubclassWindow();
423 virtual BOOL
PreTranslateMessage(MSG
* pMsg
);
424 static UINT
LogThreadEntry(LPVOID pVoid
);
426 void FetchRemoteList();
427 void FetchTrackingBranchList();
428 void FetchLastLogInfo();
429 void FetchFullLogInfo(CString
&from
, CString
&to
);
430 void FillBackGround(HDC hdc
, DWORD_PTR Index
, CRect
&rect
);
431 void DrawTagBranch(HDC hdc
, CRect
&rect
, INT_PTR index
, std::vector
<REFLABEL
> refList
);
432 void DrawGraph(HDC
,CRect
&rect
,INT_PTR index
);
434 void paintGraphLane(HDC hdc
,int laneHeight
, int type
, int x1
, int x2
,
435 const COLORREF
& col
,const COLORREF
& activeColor
, int top
) ;
436 void DrawLine(HDC hdc
, int x1
, int y1
, int x2
, int y2
){::MoveToEx(hdc
,x1
,y1
,NULL
);::LineTo(hdc
,x2
,y2
);}
438 * Save column widths to the registry
440 void SaveColumnWidths(); // save col widths to the registry
442 BOOL
IsEntryInDateRange(int i
);
446 std::vector
<GitRev
*> m_AsynDiffList
;
447 CComCriticalSection m_AsynDiffListLock
;
448 HANDLE m_AsyncDiffEvent
;
449 volatile LONG m_AsyncThreadExit
;
450 CWinThread
* m_DiffingThread
;
452 static int DiffAsync(GitRev
*rev
, void *data
)
454 ULONGLONG offset
=((CGitLogListBase
*)data
)->m_LogCache
.GetOffset(rev
->m_CommitHash
);
457 ((CGitLogListBase
*)data
)->m_AsynDiffListLock
.Lock();
458 ((CGitLogListBase
*)data
)->m_AsynDiffList
.push_back(rev
);
459 ((CGitLogListBase
*)data
)->m_AsynDiffListLock
.Unlock();
460 ::SetEvent(((CGitLogListBase
*)data
)->m_AsyncDiffEvent
);
464 if(((CGitLogListBase
*)data
)->m_LogCache
.LoadOneItem(*rev
,offset
))
466 ((CGitLogListBase
*)data
)->m_AsynDiffListLock
.Lock();
467 ((CGitLogListBase
*)data
)->m_AsynDiffList
.push_back(rev
);
468 ((CGitLogListBase
*)data
)->m_AsynDiffListLock
.Unlock();
469 ::SetEvent(((CGitLogListBase
*)data
)->m_AsyncDiffEvent
);
471 InterlockedExchange(&rev
->m_IsDiffFiles
, TRUE
);
472 if(rev
->m_IsDiffFiles
&& rev
->m_IsCommitParsed
)
473 InterlockedExchange(&rev
->m_IsFull
, TRUE
);
478 static UINT
AsyncThread(LPVOID data
)
480 return ((CGitLogListBase
*)data
)->AsyncDiffThread();
483 int AsyncDiffThread();
484 bool m_AsyncThreadExited
;
487 void SafeTerminateAsyncDiffThread()
489 if(m_DiffingThread
!=NULL
&& m_AsyncThreadExit
!= TRUE
)
491 m_AsyncThreadExit
= TRUE
;
492 ::SetEvent(m_AsyncDiffEvent
);
493 DWORD ret
= WAIT_TIMEOUT
;
494 // do not block here, but process messages and ask until the thread ends
495 while (ret
== WAIT_TIMEOUT
&& !m_AsyncThreadExited
)
497 AfxGetThread()->PumpMessage(); // process messages, so that GetTopIndex and so on in the thread work
498 ret
= ::WaitForSingleObject(m_DiffingThread
->m_hThread
, 100);
500 m_DiffingThread
= NULL
;
505 CComCriticalSection m_critSec
;
507 HICON m_hModifiedIcon
;
508 HICON m_hReplacedIcon
;
510 HICON m_hDeletedIcon
;
515 CRegDWORD m_regMaxBugIDColWidth
;
518 CStoreSelection
* m_pStoreSelection
;
522 CString m_CurrentBranch
;
525 COLORREF m_LineColors
[Lanes::COLORS_NUM
];
526 DWORD m_DateFormat
; // DATE_SHORTDATE or DATE_LONGDATE
527 bool m_bRelativeTimes
; // Show relative times
529 CString m_SingleRemote
;
530 bool m_bSymbolizeRefNames
;
531 bool m_bIncludeBoundaryCommits
;
533 ColumnManager m_ColumnManager
;
534 DWORD m_dwDefaultColumns
;