1 // RebaseDlg.cpp : implementation file
5 #include "TortoiseProc.h"
8 #include "MessageBox.h"
9 #include "UnicodeUtils.h"
10 #include "BrowseRefsDlg.h"
11 #include "ProgressDlg.h"
14 IMPLEMENT_DYNAMIC(CRebaseDlg
, CResizableStandAloneDialog
)
16 CRebaseDlg::CRebaseDlg(CWnd
* pParent
/*=NULL*/)
17 : CResizableStandAloneDialog(CRebaseDlg::IDD
, pParent
)
22 m_RebaseStage
=CHOOSE_BRANCH
;
23 m_CurrentRebaseIndex
=-1;
24 m_bThreadRunning
=FALSE
;
25 this->m_IsCherryPick
= FALSE
;
29 CRebaseDlg::~CRebaseDlg()
33 void CRebaseDlg::DoDataExchange(CDataExchange
* pDX
)
35 CDialog::DoDataExchange(pDX
);
36 DDX_Control(pDX
, IDC_REBASE_PROGRESS
, m_ProgressBar
);
37 DDX_Control(pDX
, IDC_STATUS_STATIC
, m_CtrlStatusText
);
38 DDX_Check(pDX
, IDC_PICK_ALL
, m_bPickAll
);
39 DDX_Check(pDX
, IDC_SQUASH_ALL
, m_bSquashAll
);
40 DDX_Check(pDX
, IDC_EDIT_ALL
, m_bEditAll
);
41 DDX_Control(pDX
, IDC_REBASE_SPLIT
, m_wndSplitter
);
42 DDX_Control(pDX
,IDC_COMMIT_LIST
,m_CommitList
);
43 DDX_Control(pDX
,IDC_REBASE_COMBOXEX_BRANCH
, this->m_BranchCtrl
);
44 DDX_Control(pDX
,IDC_REBASE_COMBOXEX_UPSTREAM
, this->m_UpstreamCtrl
);
45 DDX_Check(pDX
, IDC_REBASE_CHECK_FORCE
,m_bForce
);
50 BEGIN_MESSAGE_MAP(CRebaseDlg
, CResizableStandAloneDialog
)
51 ON_BN_CLICKED(IDC_PICK_ALL
, &CRebaseDlg::OnBnClickedPickAll
)
52 ON_BN_CLICKED(IDC_SQUASH_ALL
, &CRebaseDlg::OnBnClickedSquashAll
)
53 ON_BN_CLICKED(IDC_EDIT_ALL
, &CRebaseDlg::OnBnClickedEditAll
)
54 ON_BN_CLICKED(IDC_REBASE_SPLIT
, &CRebaseDlg::OnBnClickedRebaseSplit
)
55 ON_BN_CLICKED(IDC_REBASE_CONTINUE
,OnBnClickedContinue
)
56 ON_BN_CLICKED(IDC_REBASE_ABORT
, OnBnClickedAbort
)
58 ON_CBN_SELCHANGE(IDC_REBASE_COMBOXEX_BRANCH
, &CRebaseDlg::OnCbnSelchangeBranch
)
59 ON_CBN_SELCHANGE(IDC_REBASE_COMBOXEX_UPSTREAM
, &CRebaseDlg::OnCbnSelchangeUpstream
)
60 ON_MESSAGE(MSG_REBASE_UPDATE_UI
, OnRebaseUpdateUI
)
61 ON_BN_CLICKED(IDC_BUTTON_BROWSE
, &CRebaseDlg::OnBnClickedButtonBrowse
)
62 ON_BN_CLICKED(IDC_REBASE_CHECK_FORCE
, &CRebaseDlg::OnBnClickedRebaseCheckForce
)
63 ON_STN_CLICKED(IDC_STATUS_STATIC
, &CRebaseDlg::OnStnClickedStatusStatic
)
64 ON_BN_CLICKED(IDC_REBASE_POST_BUTTON
, &CRebaseDlg::OnBnClickedRebasePostButton
)
67 void CRebaseDlg::AddRebaseAnchor()
69 AddAnchor(IDC_REBASE_TAB
,TOP_LEFT
,BOTTOM_RIGHT
);
70 AddAnchor(IDC_COMMIT_LIST
,TOP_LEFT
, TOP_RIGHT
);
71 AddAnchor(IDC_REBASE_SPLIT
,TOP_LEFT
, TOP_RIGHT
);
72 AddAnchor(IDC_STATUS_STATIC
, BOTTOM_LEFT
,BOTTOM_RIGHT
);
73 AddAnchor(IDC_REBASE_CONTINUE
,BOTTOM_RIGHT
);
74 AddAnchor(IDC_REBASE_ABORT
, BOTTOM_RIGHT
);
75 AddAnchor(IDC_REBASE_PROGRESS
,BOTTOM_LEFT
, BOTTOM_RIGHT
);
76 AddAnchor(IDC_PICK_ALL
,TOP_LEFT
);
77 AddAnchor(IDC_SQUASH_ALL
,TOP_LEFT
);
78 AddAnchor(IDC_EDIT_ALL
,TOP_LEFT
);
79 AddAnchor(IDC_REBASE_COMBOXEX_UPSTREAM
,TOP_LEFT
);
80 AddAnchor(IDC_REBASE_COMBOXEX_BRANCH
,TOP_LEFT
);
81 AddAnchor(IDC_REBASE_STATIC_UPSTREAM
,TOP_LEFT
);
82 AddAnchor(IDC_REBASE_STATIC_BRANCH
,TOP_LEFT
);
83 AddAnchor(IDHELP
, BOTTOM_RIGHT
);
84 AddAnchor(IDC_REBASE_CHECK_FORCE
,TOP_RIGHT
);
85 AddAnchor(IDC_REBASE_POST_BUTTON
,BOTTOM_LEFT
);
87 this->AddOthersToAnchor();
90 BOOL
CRebaseDlg::OnInitDialog()
92 CResizableStandAloneDialog::OnInitDialog();
97 GetClientRect(m_DlgOrigRect
);
98 m_CommitList
.GetClientRect(m_CommitListOrigRect
);
100 CWnd
*pwnd
=this->GetDlgItem(IDC_REBASE_DUMY_TAB
);
101 pwnd
->GetWindowRect(&rectDummy
);
102 this->ScreenToClient(rectDummy
);
104 if (!m_ctrlTabCtrl
.Create(CMFCTabCtrl::STYLE_FLAT
, rectDummy
, this, IDC_REBASE_TAB
))
106 TRACE0("Failed to create output tab window\n");
107 return FALSE
; // fail to create
109 m_ctrlTabCtrl
.SetResizeMode(CMFCTabCtrl::RESIZE_NO
);
110 // Create output panes:
111 //const DWORD dwStyle = LBS_NOINTEGRALHEIGHT | WS_CHILD | WS_VISIBLE | WS_HSCROLL | WS_VSCROLL;
112 DWORD dwStyle
=LVS_REPORT
| LVS_SHOWSELALWAYS
| LVS_ALIGNLEFT
| WS_BORDER
| WS_TABSTOP
|LVS_SINGLESEL
|WS_CHILD
| WS_VISIBLE
;
114 if (! this->m_FileListCtrl
.Create(dwStyle
,rectDummy
,&this->m_ctrlTabCtrl
,0) )
116 TRACE0("Failed to create output windows\n");
117 return FALSE
; // fail to create
120 if( ! this->m_LogMessageCtrl
.Create(_T("Scintilla"),_T("source"),0,rectDummy
,&m_ctrlTabCtrl
,0,0) )
122 TRACE0("Failed to create log message control");
125 m_LogMessageCtrl
.Init(0);
127 dwStyle
= LBS_NOINTEGRALHEIGHT
| WS_CHILD
| WS_VISIBLE
| WS_HSCROLL
| WS_VSCROLL
;
129 if (!m_wndOutputRebase
.Create(_T("Scintilla"),_T("source"),0,rectDummy
, &m_ctrlTabCtrl
, 0,0) )
131 TRACE0("Failed to create output windows\n");
132 return -1; // fail to create
134 m_wndOutputRebase
.Init(0);
135 m_wndOutputRebase
.Call(SCI_SETREADONLY
, TRUE
);
137 m_tooltips
.Create(this);
139 m_tooltips
.AddTool(IDC_REBASE_CHECK_FORCE
,IDS_REBASE_FORCE_TT
);
140 m_tooltips
.AddTool(IDC_REBASE_ABORT
,IDS_REBASE_ABORT_TT
);
144 m_FileListCtrl
.Init(SVNSLC_COLEXT
| SVNSLC_COLSTATUS
|SVNSLC_COLADD
|SVNSLC_COLDEL
, _T("RebaseDlg"),(SVNSLC_POPALL
^ SVNSLC_POPCOMMIT
),false);
146 m_ctrlTabCtrl
.AddTab(&m_FileListCtrl
,_T("Conflict File"));
147 m_ctrlTabCtrl
.AddTab(&m_LogMessageCtrl
,_T("Commit Message"),1);
148 m_ctrlTabCtrl
.AddTab(&m_wndOutputRebase
,_T("Log"),2);
152 EnableSaveRestore(_T("RebaseDlg"));
154 DWORD yPos
= CRegDWORD(_T("Software\\TortoiseGit\\TortoiseProc\\ResizableState\\RebaseDlgSizer"));
155 RECT rcDlg
, rcLogMsg
, rcFileList
;
156 GetClientRect(&rcDlg
);
157 m_CommitList
.GetWindowRect(&rcLogMsg
);
158 ScreenToClient(&rcLogMsg
);
159 this->m_ctrlTabCtrl
.GetWindowRect(&rcFileList
);
160 ScreenToClient(&rcFileList
);
164 m_wndSplitter
.GetWindowRect(&rectSplitter
);
165 ScreenToClient(&rectSplitter
);
166 int delta
= yPos
- rectSplitter
.top
;
167 if ((rcLogMsg
.bottom
+ delta
> rcLogMsg
.top
)&&(rcLogMsg
.bottom
+ delta
< rcFileList
.bottom
- 30))
169 m_wndSplitter
.SetWindowPos(NULL
, 0, yPos
, 0, 0, SWP_NOSIZE
);
174 if( this->m_RebaseStage
== CHOOSE_BRANCH
)
176 this->LoadBranchInfo();
180 this->m_BranchCtrl
.EnableWindow(FALSE
);
181 this->m_UpstreamCtrl
.EnableWindow(FALSE
);
184 m_CommitList
.m_IsIDReplaceAction
= TRUE
;
185 // m_CommitList.m_IsOldFirst = TRUE;
186 m_CommitList
.m_IsRebaseReplaceGraph
= TRUE
;
188 m_CommitList
.InsertGitColumn();
190 this->SetControlEnable();
192 if(!this->m_PreCmd
.IsEmpty())
194 CProgressDlg progress
;
195 progress
.m_GitCmd
=m_PreCmd
;
196 progress
.m_bAutoCloseOnSuccess
=true;
202 this->m_BranchCtrl
.SetCurSel(-1);
203 this->m_BranchCtrl
.EnableWindow(FALSE
);
204 this->m_UpstreamCtrl
.EnableWindow(FALSE
);
205 this->SetWindowText(_T("Cherry Pick"));
206 this->m_CommitList
.StartFilter();
210 SetContinueButtonText();
211 m_CommitList
.DeleteAllItems();
215 m_CommitList
.m_ContextMenuMask
&= ~(m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_CHERRY_PICK
)|
216 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_SWITCHTOREV
)|
217 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_RESET
)|
218 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REVERTREV
)|
219 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_TO_VERSION
)|
220 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REVERTTOREV
)|
221 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_COMBINE_COMMIT
));
223 if(m_CommitList
.m_IsOldFirst
)
224 this->m_CurrentRebaseIndex
= -1;
226 this->m_CurrentRebaseIndex
= m_CommitList
.m_logEntries
.size();
231 // CRebaseDlg message handlers
233 void CRebaseDlg::OnBnClickedPickAll()
235 // TODO: Add your control notification handler code here
238 this->SetAllRebaseAction(CTGitPath::LOGACTIONS_REBASE_PICK
);
240 this->m_bEditAll
=FALSE
;
241 this->m_bSquashAll
=FALSE
;
242 this->UpdateData(FALSE
);
246 void CRebaseDlg::OnBnClickedSquashAll()
248 // TODO: Add your control notification handler code here
250 if(this->m_bSquashAll
)
251 this->SetAllRebaseAction(CTGitPath::LOGACTIONS_REBASE_SQUASH
);
253 this->m_bEditAll
=FALSE
;
254 this->m_bPickAll
=FALSE
;
255 this->UpdateData(FALSE
);
259 void CRebaseDlg::OnBnClickedEditAll()
261 // TODO: Add your control notification handler code here
263 if( this->m_bEditAll
)
264 this->SetAllRebaseAction(CTGitPath::LOGACTIONS_REBASE_EDIT
);
266 this->m_bPickAll
=FALSE
;
267 this->m_bSquashAll
=FALSE
;
268 this->UpdateData(FALSE
);
272 void CRebaseDlg::SetAllRebaseAction(int action
)
274 for(int i
=0;i
<this->m_CommitList
.m_logEntries
.size();i
++)
276 m_CommitList
.m_logEntries
[i
].m_Action
=action
;
278 m_CommitList
.Invalidate();
281 void CRebaseDlg::OnBnClickedRebaseSplit()
284 // TODO: Add your control notification handler code here
287 LRESULT
CRebaseDlg::DefWindowProc(UINT message
, WPARAM wParam
, LPARAM lParam
)
291 if (wParam
== IDC_REBASE_SPLIT
)
293 SPC_NMHDR
* pHdr
= (SPC_NMHDR
*) lParam
;
299 return __super::DefWindowProc(message
, wParam
, lParam
);
302 void CRebaseDlg::DoSize(int delta
)
305 this->RemoveAllAnchors();
307 CSplitterControl::ChangeHeight(GetDlgItem(IDC_COMMIT_LIST
), delta
, CW_TOPALIGN
);
308 //CSplitterControl::ChangeHeight(GetDlgItem(), delta, CW_TOPALIGN);
309 CSplitterControl::ChangeHeight(GetDlgItem(IDC_REBASE_TAB
), -delta
, CW_BOTTOMALIGN
);
310 //CSplitterControl::ChangeHeight(GetDlgItem(), -delta, CW_BOTTOMALIGN);
311 CSplitterControl::ChangePos(GetDlgItem(IDC_SQUASH_ALL
),0,delta
);
312 CSplitterControl::ChangePos(GetDlgItem(IDC_PICK_ALL
),0,delta
);
313 CSplitterControl::ChangePos(GetDlgItem(IDC_EDIT_ALL
),0,delta
);
315 this->AddRebaseAnchor();
316 // adjust the minimum size of the dialog to prevent the resizing from
317 // moving the list control too far down.
319 m_CommitList
.GetClientRect(rcLogMsg
);
320 SetMinTrackSize(CSize(m_DlgOrigRect
.Width(), m_DlgOrigRect
.Height()-m_CommitListOrigRect
.Height()+rcLogMsg
.Height()));
323 // m_CommitList.Invalidate();
325 // GetDlgItem(IDC_LOGMESSAGE)->Invalidate();
327 this->m_ctrlTabCtrl
.Invalidate();
328 this->m_CommitList
.Invalidate();
329 this->m_FileListCtrl
.Invalidate();
330 this->m_LogMessageCtrl
.Invalidate();
334 void CRebaseDlg::SetSplitterRange()
336 if ((m_CommitList
)&&(m_ctrlTabCtrl
))
339 m_CommitList
.GetWindowRect(rcTop
);
340 ScreenToClient(rcTop
);
342 m_ctrlTabCtrl
.GetWindowRect(rcMiddle
);
343 ScreenToClient(rcMiddle
);
344 if (rcMiddle
.Height() && rcMiddle
.Width())
345 m_wndSplitter
.SetRange(rcTop
.top
+60, rcMiddle
.bottom
-80);
349 void CRebaseDlg::OnSize(UINT nType
,int cx
, int cy
)
351 // first, let the resizing take place
352 __super::OnSize(nType
, cx
, cy
);
358 void CRebaseDlg::SaveSplitterPos()
362 CRegDWORD regPos
= CRegDWORD(_T("Software\\TortoiseGit\\TortoiseProc\\ResizableState\\RebaseDlgSizer"));
364 m_wndSplitter
.GetWindowRect(&rectSplitter
);
365 ScreenToClient(&rectSplitter
);
366 regPos
= rectSplitter
.top
;
370 void CRebaseDlg::LoadBranchInfo()
372 m_BranchCtrl
.SetMaxHistoryItems(0x7FFFFFFF);
373 m_UpstreamCtrl
.SetMaxHistoryItems(0x7FFFFFFF);
378 g_Git
.GetBranchList(list
,¤t
,CGit::BRANCH_ALL
);
379 m_BranchCtrl
.AddString(list
);
380 m_UpstreamCtrl
.AddString(list
);
382 m_BranchCtrl
.SetCurSel(current
);
384 AddBranchToolTips(&m_BranchCtrl
);
385 AddBranchToolTips(&m_UpstreamCtrl
);
387 if(!m_Upstream
.IsEmpty())
389 m_UpstreamCtrl
.AddString(m_Upstream
);
390 m_UpstreamCtrl
.SetCurSel(m_UpstreamCtrl
.GetCount()-1);
394 //Select pull-remote from current branch
395 CString currentBranch
= g_Git
.GetSymbolicRef();
397 configName
.Format(L
"branch.%s.remote", currentBranch
);
398 CString pullRemote
= g_Git
.GetConfigValue(configName
);
400 //Select pull-branch from current branch
401 configName
.Format(L
"branch.%s.merge", currentBranch
);
402 CString pullBranch
= CGit::StripRefName(g_Git
.GetConfigValue(configName
));
404 CString defaultUpstream
;
405 defaultUpstream
.Format(L
"remotes/%s/%s", pullRemote
, pullBranch
);
406 int found
= m_UpstreamCtrl
.FindStringExact(0, defaultUpstream
);
408 m_UpstreamCtrl
.SetCurSel(found
);
412 void CRebaseDlg::OnCbnSelchangeBranch()
417 void CRebaseDlg::OnCbnSelchangeUpstream()
422 void CRebaseDlg::FetchLogList()
428 cmd
.Format(_T("git.exe merge-base %s %s"), m_UpstreamCtrl
.GetString(),m_BranchCtrl
.GetString());
429 if(g_Git
.Run(cmd
,&base
,CP_ACP
))
431 CMessageBox::Show(NULL
,base
,_T("TortoiseGit"),MB_OK
|MB_ICONERROR
);
436 cmd
.Format(_T("git.exe rev-parse %s"), m_UpstreamCtrl
.GetString());
437 if( g_Git
.Run(cmd
,&hash
,CP_ACP
))
439 CMessageBox::Show(NULL
,base
,_T("TortoiseGit"),MB_OK
|MB_ICONERROR
);
446 m_CommitList
.Clear();
448 fmt
.LoadString(IDS_REBASE_UPTODATE_FMT
);
449 text
.Format(fmt
,m_BranchCtrl
.GetString());
450 m_CommitList
.ShowText(text
);
451 this->GetDlgItem(IDC_REBASE_CONTINUE
)->EnableWindow(m_CommitList
.GetItemCount());
456 m_CommitList
.Clear();
457 this->m_CommitList
.FillGitLog(NULL
,0,&m_UpstreamCtrl
.GetString(),&m_BranchCtrl
.GetString());
458 if( m_CommitList
.GetItemCount() == 0 )
459 m_CommitList
.ShowText(_T("Nothing to Rebase"));
461 CString hash
=g_Git
.GetHash(m_UpstreamCtrl
.GetString());
464 if(m_CommitList
.m_logEntries
[m_CommitList
.m_logEntries
.size()-1].m_ParentHash
.size() >=0 )
466 if(hash
== m_CommitList
.m_logEntries
[m_CommitList
.m_logEntries
.size()-1].m_ParentHash
[0])
468 m_CommitList
.Clear();
469 m_CommitList
.ShowText(_T("Nothing Rebase"));
475 AddBranchToolTips(&this->m_BranchCtrl
);
476 AddBranchToolTips(&this->m_UpstreamCtrl
);
478 for(int i
=0;i
<m_CommitList
.m_logEntries
.size();i
++)
480 m_CommitList
.m_logEntries
[i
].m_Action
= CTGitPath::LOGACTIONS_REBASE_PICK
;
483 m_CommitList
.Invalidate();
485 if(m_CommitList
.m_IsOldFirst
)
486 this->m_CurrentRebaseIndex
= -1;
488 this->m_CurrentRebaseIndex
= m_CommitList
.m_logEntries
.size();
490 this->GetDlgItem(IDC_REBASE_CONTINUE
)->EnableWindow(m_CommitList
.GetItemCount());
493 void CRebaseDlg::AddBranchToolTips(CHistoryCombo
*pBranch
)
497 CString text
=pBranch
->GetString();
500 g_Git
.GetLog(data
,text
,NULL
,1,0);
502 rev
.ParserFromLog(data
);
503 tooltip
.Format(_T("CommitHash:%s\nCommit by: %s %s\n <b>%s</b> \n %s"),
506 CAppUtils::FormatDateAndTime(rev
.m_AuthorDate
,DATE_LONGDATE
),
510 pBranch
->DisableTooltip();
511 this->m_tooltips
.AddTool(pBranch
->GetComboBoxCtrl(),tooltip
);
515 BOOL
CRebaseDlg::PreTranslateMessage(MSG
*pMsg
)
517 m_tooltips
.RelayEvent(pMsg
);
518 return CResizableStandAloneDialog::PreTranslateMessage(pMsg
);
520 int CRebaseDlg::CheckRebaseCondition()
522 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_LOG
);
524 if( !g_Git
.CheckCleanWorkTree() )
526 CMessageBox::Show(NULL
,_T("Rebase Need Clean Working Tree"),_T("TortoiseGit"),MB_OK
);
529 //Todo Check $REBASE_ROOT
533 cmd
=_T("git.exe var GIT_COMMITTER_IDENT");
534 if(g_Git
.Run(cmd
,NULL
,CP_UTF8
))
537 //Todo call pre_rebase_hook
540 int CRebaseDlg::StartRebase()
544 if(!this->m_IsCherryPick
)
546 //Todo call comment_for_reflog
547 cmd
.Format(_T("git.exe checkout %s"),this->m_BranchCtrl
.GetString());
548 this->AddLogString(cmd
);
550 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
553 this->AddLogString(out
);
556 cmd
=_T("git.exe rev-parse --verify HEAD");
557 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
559 AddLogString(_T("No Head"));
563 //git symbolic-ref HEAD > "$DOTEST"/head-name 2> /dev/null ||
564 // echo "detached HEAD" > "$DOTEST"/head-name
566 cmd
.Format(_T("git.exe update-ref ORIG_HEAD HEAD"));
567 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
569 AddLogString(_T("update ORIG_HEAD Fail"));
573 if( !this->m_IsCherryPick
)
575 cmd
.Format(_T("git.exe checkout %s"),this->m_UpstreamCtrl
.GetString());
576 this->AddLogString(cmd
);
579 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
585 m_OrigUpstreamHash
.Empty();
586 m_OrigUpstreamHash
= g_Git
.GetHash(this->m_UpstreamCtrl
.GetString());
587 if(m_OrigUpstreamHash
.IsEmpty())
589 this->AddLogString(m_OrigUpstreamHash
);
593 if( !this->m_IsCherryPick
)
595 cmd
.Format(_T("git.exe rev-parse %s"),this->m_BranchCtrl
.GetString());
596 if(g_Git
.Run(cmd
,&this->m_OrigBranchHash
,CP_UTF8
))
598 this->AddLogString(m_OrigBranchHash
);
601 this->AddLogString(_T("Start Rebase\r\n"));
604 this->AddLogString(_T("Start Cherry-pick\r\n"));
608 int CRebaseDlg::VerifyNoConflict()
611 if(g_Git
.ListConflictFile(list
))
613 AddLogString(_T("Get conflict files fail"));
616 if( list
.GetCount() != 0 )
618 CMessageBox::Show(NULL
,_T("There are conflict file, you should mark it resolve"),_T("TortoiseGit"),MB_OK
);
624 int CRebaseDlg::FinishRebase()
626 if(this->m_IsCherryPick
) //cherry pick mode no "branch", working at upstream branch
630 cmd
.Format(_T("git.exe branch -f %s"),this->m_BranchCtrl
.GetString());
631 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
637 cmd
.Format(_T("git.exe reset --hard %s"),this->m_OrigUpstreamHash
);
638 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
644 cmd
.Format(_T("git.exe checkout -f %s"),this->m_BranchCtrl
.GetString());
645 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
652 void CRebaseDlg::OnBnClickedContinue()
654 if( m_RebaseStage
== CHOOSE_BRANCH
|| m_RebaseStage
== CHOOSE_COMMIT_PICK_MODE
)
656 if(CheckRebaseCondition())
658 m_RebaseStage
= REBASE_START
;
661 if( m_RebaseStage
== REBASE_DONE
)
666 if( m_RebaseStage
== REBASE_FINISH
)
674 if( m_RebaseStage
== REBASE_SQUASH_CONFLICT
)
676 if(VerifyNoConflict())
678 GitRev
*curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
679 if(this->CheckNextCommitIsSquash())
680 {//next commit is not squash;
681 m_RebaseStage
= REBASE_SQUASH_EDIT
;
682 this->OnRebaseUpdateUI(0,0);
683 this->UpdateCurrentStatus();
687 m_RebaseStage
=REBASE_CONTINUE
;
688 curRev
->m_Action
|=CTGitPath::LOGACTIONS_REBASE_DONE
;
689 this->UpdateCurrentStatus();
693 if( m_RebaseStage
== REBASE_CONFLICT
)
695 if(VerifyNoConflict())
698 GitRev
*curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
702 cmd
.Format(_T("git.exe commit -C %s"), curRev
->m_CommitHash
);
704 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
706 if(!g_Git
.CheckCleanWorkTree())
708 CMessageBox::Show(NULL
,out
,_T("TortoiseGit"),MB_OK
|MB_ICONERROR
);
714 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_LOG
);
715 if( curRev
->m_Action
& CTGitPath::LOGACTIONS_REBASE_EDIT
)
717 m_RebaseStage
=REBASE_EDIT
;
718 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_MESSAGE
);
719 this->UpdateCurrentStatus();
724 m_RebaseStage
=REBASE_CONTINUE
;
725 curRev
->m_Action
|=CTGitPath::LOGACTIONS_REBASE_DONE
;
726 this->UpdateCurrentStatus();
731 if( m_RebaseStage
== REBASE_EDIT
|| m_RebaseStage
== REBASE_SQUASH_EDIT
)
734 GitRev
*curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
736 str
=this->m_LogMessageCtrl
.GetText();
737 if(str
.Trim().IsEmpty())
739 CMessageBox::Show(NULL
,_T("Commit Message Is Empty"),_T("TortoiseGit"),MB_OK
|MB_ICONERROR
);
743 CString tempfile
=::GetTempFile();
744 CFile
file(tempfile
,CFile::modeReadWrite
|CFile::modeCreate
);
745 CStringA log
=CUnicodeUtils::GetUTF8( str
);
746 file
.Write(log
,log
.GetLength());
747 //file.WriteString(m_sLogMessage);
752 if( m_RebaseStage
== REBASE_SQUASH_EDIT
)
753 cmd
.Format(_T("git.exe commit -F \"%s\""), tempfile
);
755 cmd
.Format(_T("git.exe commit --amend -F \"%s\""), tempfile
);
757 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
759 if(!g_Git
.CheckCleanWorkTree())
761 CMessageBox::Show(NULL
,out
,_T("TortoiseGit"),MB_OK
|MB_ICONERROR
);
766 CFile::Remove(tempfile
);
768 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_LOG
);
769 m_RebaseStage
=REBASE_CONTINUE
;
770 curRev
->m_Action
|=CTGitPath::LOGACTIONS_REBASE_DONE
;
771 this->UpdateCurrentStatus();
775 InterlockedExchange(&m_bThreadRunning
, TRUE
);
778 if (AfxBeginThread(RebaseThreadEntry
, this)==NULL
)
780 InterlockedExchange(&m_bThreadRunning
, FALSE
);
781 CMessageBox::Show(NULL
, _T("Create Rebase Thread Fail"), _T("TortoiseGit"), MB_OK
| MB_ICONERROR
);
785 int CRebaseDlg::CheckNextCommitIsSquash()
788 if(m_CommitList
.m_IsOldFirst
)
789 index
=m_CurrentRebaseIndex
+1;
791 index
=m_CurrentRebaseIndex
-1;
798 if(index
>= m_CommitList
.GetItemCount())
801 curRev
=(GitRev
*)m_CommitList
.m_arShownList
[index
];
803 if( curRev
->m_Action
&CTGitPath::LOGACTIONS_REBASE_SQUASH
)
805 if( curRev
->m_Action
&CTGitPath::LOGACTIONS_REBASE_SKIP
)
807 if(m_CommitList
.m_IsOldFirst
)
814 }while(curRev
->m_Action
&CTGitPath::LOGACTIONS_REBASE_SKIP
);
819 int CRebaseDlg::GoNext()
821 if(m_CommitList
.m_IsOldFirst
)
822 m_CurrentRebaseIndex
++;
824 m_CurrentRebaseIndex
--;
828 int CRebaseDlg::StateAction()
830 switch(this->m_RebaseStage
)
833 case CHOOSE_COMMIT_PICK_MODE
:
836 m_RebaseStage
= REBASE_START
;
843 void CRebaseDlg::SetContinueButtonText()
846 switch(this->m_RebaseStage
)
849 case CHOOSE_COMMIT_PICK_MODE
:
854 case REBASE_CONTINUE
:
855 case REBASE_SQUASH_CONFLICT
:
856 Text
= _T("Continue");
859 case REBASE_CONFLICT
:
866 case REBASE_SQUASH_EDIT
:
879 this->GetDlgItem(IDC_REBASE_CONTINUE
)->SetWindowText(Text
);
882 void CRebaseDlg::SetControlEnable()
884 switch(this->m_RebaseStage
)
887 case CHOOSE_COMMIT_PICK_MODE
:
889 this->GetDlgItem(IDC_PICK_ALL
)->EnableWindow(TRUE
);
890 this->GetDlgItem(IDC_EDIT_ALL
)->EnableWindow(TRUE
);
891 this->GetDlgItem(IDC_SQUASH_ALL
)->EnableWindow(TRUE
);
894 this->GetDlgItem(IDC_REBASE_COMBOXEX_BRANCH
)->EnableWindow(TRUE
);
895 this->GetDlgItem(IDC_REBASE_COMBOXEX_UPSTREAM
)->EnableWindow(TRUE
);
897 //this->m_CommitList.m_IsEnableRebaseMenu=TRUE;
898 this->m_CommitList
.m_ContextMenuMask
|= m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_PICK
)|
899 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_SQUASH
)|
900 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_EDIT
)|
901 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_SKIP
);
905 case REBASE_CONTINUE
:
908 case REBASE_CONFLICT
:
910 case REBASE_SQUASH_CONFLICT
:
912 this->GetDlgItem(IDC_PICK_ALL
)->EnableWindow(FALSE
);
913 this->GetDlgItem(IDC_EDIT_ALL
)->EnableWindow(FALSE
);
914 this->GetDlgItem(IDC_SQUASH_ALL
)->EnableWindow(FALSE
);
915 this->GetDlgItem(IDC_REBASE_COMBOXEX_BRANCH
)->EnableWindow(FALSE
);
916 this->GetDlgItem(IDC_REBASE_COMBOXEX_UPSTREAM
)->EnableWindow(FALSE
);
917 //this->m_CommitList.m_IsEnableRebaseMenu=FALSE;
918 this->m_CommitList
.m_ContextMenuMask
&= ~(m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_PICK
)|
919 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_SQUASH
)|
920 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_EDIT
)|
921 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_SKIP
));
923 if( m_RebaseStage
== REBASE_DONE
&& (!this->m_PostButtonText
.IsEmpty()) )
925 this->GetDlgItem(IDC_STATUS_STATIC
)->ShowWindow(SW_HIDE
);
926 this->GetDlgItem(IDC_REBASE_POST_BUTTON
)->ShowWindow(SW_SHOWNORMAL
);
927 this->GetDlgItem(IDC_REBASE_POST_BUTTON
)->SetWindowText(this->m_PostButtonText
);
934 this->GetDlgItem(IDC_REBASE_CONTINUE
)->EnableWindow(FALSE
);
935 this->GetDlgItem(IDC_REBASE_ABORT
)->EnableWindow(FALSE
);
939 this->GetDlgItem(IDC_REBASE_CONTINUE
)->EnableWindow(TRUE
);
940 this->GetDlgItem(IDC_REBASE_ABORT
)->EnableWindow(TRUE
);
944 void CRebaseDlg::UpdateProgress()
949 if(m_CommitList
.m_IsOldFirst
)
950 index
= m_CurrentRebaseIndex
+1;
952 index
= m_CommitList
.GetItemCount()-m_CurrentRebaseIndex
;
954 m_ProgressBar
.SetRange(1,m_CommitList
.GetItemCount());
955 m_ProgressBar
.SetPos(index
);
957 if(m_CurrentRebaseIndex
>=0 && m_CurrentRebaseIndex
< m_CommitList
.GetItemCount())
960 text
.Format(_T("Rebasing...(%d/%d)"),index
,m_CommitList
.GetItemCount());
961 m_CtrlStatusText
.SetWindowText(text
);
965 GitRev
*prevRev
=NULL
, *curRev
=NULL
;
967 if( m_CurrentRebaseIndex
>= 0 && m_CurrentRebaseIndex
< m_CommitList
.m_arShownList
.GetSize())
969 curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
972 for(int i
=0;i
<m_CommitList
.m_arShownList
.GetSize();i
++)
974 prevRev
=(GitRev
*)m_CommitList
.m_arShownList
[i
];
975 if(prevRev
->m_Action
& CTGitPath::LOGACTIONS_REBASE_CURRENT
)
977 prevRev
->m_Action
&= ~ CTGitPath::LOGACTIONS_REBASE_CURRENT
;
978 m_CommitList
.GetItemRect(i
,&rect
,LVIR_BOUNDS
);
979 m_CommitList
.InvalidateRect(rect
);
985 curRev
->m_Action
|= CTGitPath::LOGACTIONS_REBASE_CURRENT
;
986 m_CommitList
.GetItemRect(m_CurrentRebaseIndex
,&rect
,LVIR_BOUNDS
);
987 m_CommitList
.InvalidateRect(rect
);
989 m_CommitList
.EnsureVisible(m_CurrentRebaseIndex
,FALSE
);
993 void CRebaseDlg::UpdateCurrentStatus()
995 if( m_CurrentRebaseIndex
< 0 && m_RebaseStage
!= REBASE_DONE
)
997 if(m_CommitList
.m_IsOldFirst
)
998 m_RebaseStage
= CRebaseDlg::REBASE_START
;
1000 m_RebaseStage
= CRebaseDlg::REBASE_FINISH
;
1003 if( m_CurrentRebaseIndex
== m_CommitList
.m_arShownList
.GetSize() && m_RebaseStage
!= REBASE_DONE
)
1005 if(m_CommitList
.m_IsOldFirst
)
1006 m_RebaseStage
= CRebaseDlg::REBASE_DONE
;
1008 m_RebaseStage
= CRebaseDlg::REBASE_FINISH
;
1011 SetContinueButtonText();
1016 void CRebaseDlg::AddLogString(CString str
)
1018 this->m_wndOutputRebase
.SendMessage(SCI_SETREADONLY
, FALSE
);
1019 CStringA sTextA
= m_wndOutputRebase
.StringForControl(str
);//CUnicodeUtils::GetUTF8(str);
1020 this->m_wndOutputRebase
.SendMessage(SCI_REPLACESEL
, 0, (LPARAM
)(LPCSTR
)sTextA
);
1021 this->m_wndOutputRebase
.SendMessage(SCI_REPLACESEL
, 0, (LPARAM
)(LPCSTR
)"\n");
1022 this->m_wndOutputRebase
.SendMessage(SCI_SETREADONLY
, TRUE
);
1025 int CRebaseDlg::GetCurrentCommitID()
1027 if(m_CommitList
.m_IsOldFirst
)
1029 return this->m_CurrentRebaseIndex
+1;
1033 return m_CommitList
.GetItemCount()-m_CurrentRebaseIndex
;
1037 int CRebaseDlg::DoRebase()
1040 if(m_CurrentRebaseIndex
<0)
1042 if(m_CurrentRebaseIndex
>= m_CommitList
.GetItemCount() )
1045 GitRev
*pRev
= (GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
1046 int mode
=pRev
->m_Action
& CTGitPath::LOGACTIONS_REBASE_MODE_MASK
;
1049 if( mode
== CTGitPath::LOGACTIONS_REBASE_SKIP
)
1051 pRev
->m_Action
|= CTGitPath::LOGACTIONS_REBASE_DONE
;
1055 if( mode
!= CTGitPath::LOGACTIONS_REBASE_PICK
)
1057 this->m_SquashMessage
+= pRev
->m_Subject
;
1058 this->m_SquashMessage
+= _T("\n");
1059 this->m_SquashMessage
+= pRev
->m_Body
;
1062 this->m_SquashMessage
.Empty();
1064 if(mode
== CTGitPath::LOGACTIONS_REBASE_SQUASH
)
1065 nocommit
=_T(" --no-commit ");
1068 log
.Format(_T("%s %d:%s"),CTGitPath::GetActionName(mode
),this->GetCurrentCommitID(),pRev
->m_CommitHash
);
1070 AddLogString(pRev
->m_Subject
);
1071 cmd
.Format(_T("git.exe cherry-pick %s %s"),nocommit
,pRev
->m_CommitHash
);
1073 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1077 if(g_Git
.ListConflictFile(list
))
1079 AddLogString(_T("Get conflict files fail"));
1082 if(list
.GetCount() == 0 )
1084 if(mode
== CTGitPath::LOGACTIONS_REBASE_PICK
)
1086 pRev
->m_Action
|= CTGitPath::LOGACTIONS_REBASE_DONE
;
1089 if(mode
== CTGitPath::LOGACTIONS_REBASE_EDIT
)
1091 this->m_RebaseStage
= REBASE_EDIT
;
1092 return -1; // Edit return -1 to stop rebase.
1095 if(CheckNextCommitIsSquash())
1097 // let user edit last commmit message
1098 this->m_RebaseStage
= REBASE_SQUASH_EDIT
;
1102 if(mode
== CTGitPath::LOGACTIONS_REBASE_SQUASH
)
1103 m_RebaseStage
= REBASE_SQUASH_CONFLICT
;
1105 m_RebaseStage
= REBASE_CONFLICT
;
1111 if(mode
== CTGitPath::LOGACTIONS_REBASE_PICK
)
1113 pRev
->m_Action
|= CTGitPath::LOGACTIONS_REBASE_DONE
;
1116 if(mode
== CTGitPath::LOGACTIONS_REBASE_EDIT
)
1118 this->m_RebaseStage
= REBASE_EDIT
;
1119 return -1; // Edit return -1 to stop rebase.
1123 if(CheckNextCommitIsSquash())
1125 // let user edit last commmit message
1126 this->m_RebaseStage
= REBASE_SQUASH_EDIT
;
1134 BOOL
CRebaseDlg::IsEnd()
1136 if(m_CommitList
.m_IsOldFirst
)
1137 return m_CurrentRebaseIndex
>= this->m_CommitList
.GetItemCount();
1139 return m_CurrentRebaseIndex
<0;
1142 int CRebaseDlg::RebaseThread()
1147 if( m_RebaseStage
== REBASE_START
)
1149 if( this->StartRebase() )
1151 InterlockedExchange(&m_bThreadRunning
, FALSE
);
1155 m_RebaseStage
= REBASE_CONTINUE
;
1157 }else if( m_RebaseStage
== REBASE_CONTINUE
)
1163 m_RebaseStage
= REBASE_FINISH
;
1175 }else if( m_RebaseStage
== REBASE_FINISH
)
1178 m_RebaseStage
= REBASE_DONE
;
1185 this->PostMessage(MSG_REBASE_UPDATE_UI
);
1186 //this->UpdateCurrentStatus();
1189 InterlockedExchange(&m_bThreadRunning
, FALSE
);
1190 this->PostMessage(MSG_REBASE_UPDATE_UI
);
1194 void CRebaseDlg::ListConflictFile()
1196 this->m_FileListCtrl
.Clear();
1201 this->m_FileListCtrl
.GetStatus(&list
,true);
1202 this->m_FileListCtrl
.Show(CTGitPath::LOGACTIONS_UNMERGED
|CTGitPath::LOGACTIONS_MODIFIED
|CTGitPath::LOGACTIONS_ADDED
|CTGitPath::LOGACTIONS_DELETED
,
1203 CTGitPath::LOGACTIONS_UNMERGED
);
1204 if( this->m_FileListCtrl
.GetItemCount() == 0 )
1210 LRESULT
CRebaseDlg::OnRebaseUpdateUI(WPARAM
,LPARAM
)
1212 UpdateCurrentStatus();
1213 if(m_CurrentRebaseIndex
<0)
1215 if(m_CurrentRebaseIndex
>= m_CommitList
.GetItemCount() )
1217 GitRev
*curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
1219 switch(m_RebaseStage
)
1221 case REBASE_CONFLICT
:
1222 case REBASE_SQUASH_CONFLICT
:
1224 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_CONFLICT
);
1225 this->m_LogMessageCtrl
.SetText(curRev
->m_Subject
+_T("\n")+curRev
->m_Body
);
1228 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_MESSAGE
);
1229 this->m_LogMessageCtrl
.SetText(curRev
->m_Subject
+_T("\n")+curRev
->m_Body
);
1231 case REBASE_SQUASH_EDIT
:
1232 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_MESSAGE
);
1233 this->m_LogMessageCtrl
.SetText(this->m_SquashMessage
);
1236 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_LOG
);
1240 void CRebaseDlg::OnCancel()
1244 void CRebaseDlg::OnBnClickedAbort()
1247 if(m_OrigUpstreamHash
.IsEmpty())
1249 __super::OnCancel();
1252 if(m_RebaseStage
== CHOOSE_BRANCH
|| m_RebaseStage
== CHOOSE_COMMIT_PICK_MODE
)
1257 if(CMessageBox::Show(NULL
,_T("Are you sure you want to abort the rebase process?"),_T("TortoiseGit"),MB_YESNO
) != IDYES
)
1260 cmd
.Format(_T("git.exe checkout -f %s"),this->m_UpstreamCtrl
.GetString());
1261 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1267 cmd
.Format(_T("git.exe reset --hard %s"),this->m_OrigUpstreamHash
.Left(40));
1268 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1274 if(this->m_IsCherryPick
) //there are not "branch" at cherry pick mode
1277 cmd
.Format(_T("git checkout -f %s"),this->m_BranchCtrl
.GetString());
1278 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1284 cmd
.Format(_T("git.exe reset --hard %s"),this->m_OrigBranchHash
.Left(40));
1285 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1290 __super::OnCancel();
1293 void CRebaseDlg::OnBnClickedButtonBrowse()
1295 if(CBrowseRefsDlg::PickRefForCombo(&m_UpstreamCtrl
, gPickRef_NoTag
))
1296 OnCbnSelchangeUpstream();
1299 void CRebaseDlg::OnBnClickedRebaseCheckForce()
1301 // TODO: Add your control notification handler code here
1303 this->FetchLogList();
1306 void CRebaseDlg::OnStnClickedStatusStatic()
1308 // TODO: Add your control notification handler code here
1311 void CRebaseDlg::OnBnClickedRebasePostButton()
1313 // TODO: Add your control notification handler code here
1314 this->m_Upstream
=this->m_UpstreamCtrl
.GetString();
1315 this->m_Branch
=this->m_BranchCtrl
.GetString();
1317 this->EndDialog(IDC_REBASE_POST_BUTTON
);