1 // RebaseDlg.cpp : implementation file
5 #include "TortoiseProc.h"
8 #include "MessageBox.h"
9 #include "UnicodeUtils.h"
12 IMPLEMENT_DYNAMIC(CRebaseDlg
, CResizableStandAloneDialog
)
14 CRebaseDlg::CRebaseDlg(CWnd
* pParent
/*=NULL*/)
15 : CResizableStandAloneDialog(CRebaseDlg::IDD
, pParent
)
20 m_RebaseStage
=CHOOSE_BRANCH
;
21 m_CurrentRebaseIndex
=-1;
22 m_bThreadRunning
=FALSE
;
23 this->m_IsCherryPick
= FALSE
;
26 CRebaseDlg::~CRebaseDlg()
30 void CRebaseDlg::DoDataExchange(CDataExchange
* pDX
)
32 CDialog::DoDataExchange(pDX
);
33 DDX_Control(pDX
, IDC_REBASE_PROGRESS
, m_ProgressBar
);
34 DDX_Control(pDX
, IDC_STATUS_STATIC
, m_CtrlStatusText
);
35 DDX_Check(pDX
, IDC_PICK_ALL
, m_bPickAll
);
36 DDX_Check(pDX
, IDC_SQUASH_ALL
, m_bSquashAll
);
37 DDX_Check(pDX
, IDC_EDIT_ALL
, m_bEditAll
);
38 DDX_Control(pDX
, IDC_REBASE_SPLIT
, m_wndSplitter
);
39 DDX_Control(pDX
,IDC_COMMIT_LIST
,m_CommitList
);
40 DDX_Control(pDX
,IDC_REBASE_COMBOXEX_BRANCH
, this->m_BranchCtrl
);
41 DDX_Control(pDX
,IDC_REBASE_COMBOXEX_UPSTREAM
, this->m_UpstreamCtrl
);
46 BEGIN_MESSAGE_MAP(CRebaseDlg
, CResizableStandAloneDialog
)
47 ON_BN_CLICKED(IDC_PICK_ALL
, &CRebaseDlg::OnBnClickedPickAll
)
48 ON_BN_CLICKED(IDC_SQUASH_ALL
, &CRebaseDlg::OnBnClickedSquashAll
)
49 ON_BN_CLICKED(IDC_EDIT_ALL
, &CRebaseDlg::OnBnClickedEditAll
)
50 ON_BN_CLICKED(IDC_REBASE_SPLIT
, &CRebaseDlg::OnBnClickedRebaseSplit
)
51 ON_BN_CLICKED(IDC_REBASE_CONTINUE
,OnBnClickedContinue
)
52 ON_BN_CLICKED(IDC_REBASE_ABORT
, OnBnClickedAbort
)
54 ON_CBN_SELCHANGE(IDC_REBASE_COMBOXEX_BRANCH
, &CRebaseDlg::OnCbnSelchangeBranch
)
55 ON_CBN_SELCHANGE(IDC_REBASE_COMBOXEX_UPSTREAM
, &CRebaseDlg::OnCbnSelchangeUpstream
)
56 ON_MESSAGE(MSG_REBASE_UPDATE_UI
, OnRebaseUpdateUI
)
59 void CRebaseDlg::AddRebaseAnchor()
61 AddAnchor(IDC_REBASE_TAB
,TOP_LEFT
,BOTTOM_RIGHT
);
62 AddAnchor(IDC_COMMIT_LIST
,TOP_LEFT
, TOP_RIGHT
);
63 AddAnchor(IDC_REBASE_SPLIT
,TOP_LEFT
, TOP_RIGHT
);
64 AddAnchor(IDC_STATUS_STATIC
, BOTTOM_LEFT
,BOTTOM_RIGHT
);
65 AddAnchor(IDC_REBASE_CONTINUE
,BOTTOM_RIGHT
);
66 AddAnchor(IDC_REBASE_ABORT
, BOTTOM_RIGHT
);
67 AddAnchor(IDC_REBASE_PROGRESS
,BOTTOM_LEFT
, BOTTOM_RIGHT
);
68 AddAnchor(IDC_PICK_ALL
,TOP_LEFT
);
69 AddAnchor(IDC_SQUASH_ALL
,TOP_LEFT
);
70 AddAnchor(IDC_EDIT_ALL
,TOP_LEFT
);
71 AddAnchor(IDC_REBASE_COMBOXEX_UPSTREAM
,TOP_LEFT
);
72 AddAnchor(IDC_REBASE_COMBOXEX_BRANCH
,TOP_LEFT
);
73 AddAnchor(IDC_REBASE_STATIC_UPSTREAM
,TOP_LEFT
);
74 AddAnchor(IDC_REBASE_STATIC_BRANCH
,TOP_LEFT
);
75 this->AddOthersToAnchor();
78 BOOL
CRebaseDlg::OnInitDialog()
80 CResizableStandAloneDialog::OnInitDialog();
85 GetClientRect(m_DlgOrigRect
);
86 m_CommitList
.GetClientRect(m_CommitListOrigRect
);
88 CWnd
*pwnd
=this->GetDlgItem(IDC_REBASE_DUMY_TAB
);
89 pwnd
->GetWindowRect(&rectDummy
);
90 this->ScreenToClient(rectDummy
);
92 if (!m_ctrlTabCtrl
.Create(CMFCTabCtrl::STYLE_FLAT
, rectDummy
, this, IDC_REBASE_TAB
))
94 TRACE0("Failed to create output tab window\n");
95 return FALSE
; // fail to create
97 m_ctrlTabCtrl
.SetResizeMode(CMFCTabCtrl::RESIZE_NO
);
98 // Create output panes:
99 //const DWORD dwStyle = LBS_NOINTEGRALHEIGHT | WS_CHILD | WS_VISIBLE | WS_HSCROLL | WS_VSCROLL;
100 DWORD dwStyle
=LVS_REPORT
| LVS_SHOWSELALWAYS
| LVS_ALIGNLEFT
| WS_BORDER
| WS_TABSTOP
|LVS_SINGLESEL
|WS_CHILD
| WS_VISIBLE
;
102 if (! this->m_FileListCtrl
.Create(dwStyle
,rectDummy
,&this->m_ctrlTabCtrl
,0) )
104 TRACE0("Failed to create output windows\n");
105 return FALSE
; // fail to create
108 if( ! this->m_LogMessageCtrl
.Create(_T("Scintilla"),_T("source"),0,rectDummy
,&m_ctrlTabCtrl
,0,0) )
110 TRACE0("Failed to create log message control");
113 m_LogMessageCtrl
.Init(0);
115 dwStyle
= LBS_NOINTEGRALHEIGHT
| WS_CHILD
| WS_VISIBLE
| WS_HSCROLL
| WS_VSCROLL
;
117 if (!m_wndOutputRebase
.Create(_T("Scintilla"),_T("source"),0,rectDummy
, &m_ctrlTabCtrl
, 0,0) )
119 TRACE0("Failed to create output windows\n");
120 return -1; // fail to create
122 m_wndOutputRebase
.Init(0);
123 m_wndOutputRebase
.Call(SCI_SETREADONLY
, TRUE
);
125 m_tooltips
.Create(this);
127 m_FileListCtrl
.Init(SVNSLC_COLEXT
| SVNSLC_COLSTATUS
|SVNSLC_COLADD
|SVNSLC_COLDEL
, _T("RebaseDlg"),(SVNSLC_POPALL
^ SVNSLC_POPCOMMIT
),false);
129 m_ctrlTabCtrl
.AddTab(&m_FileListCtrl
,_T("Conflict File"));
130 m_ctrlTabCtrl
.AddTab(&m_LogMessageCtrl
,_T("Commit Message"),1);
131 m_ctrlTabCtrl
.AddTab(&m_wndOutputRebase
,_T("Log"),2);
135 EnableSaveRestore(_T("RebaseDlg"));
137 DWORD yPos
= CRegDWORD(_T("Software\\TortoiseGit\\TortoiseProc\\ResizableState\\RebaseDlgSizer"));
138 RECT rcDlg
, rcLogMsg
, rcFileList
;
139 GetClientRect(&rcDlg
);
140 m_CommitList
.GetWindowRect(&rcLogMsg
);
141 ScreenToClient(&rcLogMsg
);
142 this->m_ctrlTabCtrl
.GetWindowRect(&rcFileList
);
143 ScreenToClient(&rcFileList
);
147 m_wndSplitter
.GetWindowRect(&rectSplitter
);
148 ScreenToClient(&rectSplitter
);
149 int delta
= yPos
- rectSplitter
.top
;
150 if ((rcLogMsg
.bottom
+ delta
> rcLogMsg
.top
)&&(rcLogMsg
.bottom
+ delta
< rcFileList
.bottom
- 30))
152 m_wndSplitter
.SetWindowPos(NULL
, 0, yPos
, 0, 0, SWP_NOSIZE
);
157 if( this->m_RebaseStage
== CHOOSE_BRANCH
)
159 this->LoadBranchInfo();
163 this->m_BranchCtrl
.EnableWindow(FALSE
);
164 this->m_UpstreamCtrl
.EnableWindow(FALSE
);
167 m_CommitList
.m_IsIDReplaceAction
= TRUE
;
168 // m_CommitList.m_IsOldFirst = TRUE;
169 m_CommitList
.m_IsRebaseReplaceGraph
= TRUE
;
171 m_CommitList
.InsertGitColumn();
173 this->SetControlEnable();
177 this->m_BranchCtrl
.SetCurSel(-1);
178 this->m_BranchCtrl
.EnableWindow(FALSE
);
179 this->m_UpstreamCtrl
.EnableWindow(FALSE
);
180 this->SetWindowText(_T("Cherry Pick"));
181 this->m_CommitList
.StartFilter();
185 SetContinueButtonText();
186 m_CommitList
.DeleteAllItems();
190 m_CommitList
.m_ContextMenuMask
&= ~(m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_CHERRY_PICK
)|
191 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_SWITCHTOREV
)|
192 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_RESET
)|
193 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REVERTREV
)|
194 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_TO_VERSION
)|
195 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REVERTTOREV
)|
196 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_COMBINE_COMMIT
));
198 if(m_CommitList
.m_IsOldFirst
)
199 this->m_CurrentRebaseIndex
= -1;
201 this->m_CurrentRebaseIndex
= m_CommitList
.m_logEntries
.size();
206 // CRebaseDlg message handlers
208 void CRebaseDlg::OnBnClickedPickAll()
210 // TODO: Add your control notification handler code here
213 this->SetAllRebaseAction(CTGitPath::LOGACTIONS_REBASE_PICK
);
215 this->m_bEditAll
=FALSE
;
216 this->m_bSquashAll
=FALSE
;
217 this->UpdateData(FALSE
);
221 void CRebaseDlg::OnBnClickedSquashAll()
223 // TODO: Add your control notification handler code here
225 if(this->m_bSquashAll
)
226 this->SetAllRebaseAction(CTGitPath::LOGACTIONS_REBASE_SQUASH
);
228 this->m_bEditAll
=FALSE
;
229 this->m_bPickAll
=FALSE
;
230 this->UpdateData(FALSE
);
234 void CRebaseDlg::OnBnClickedEditAll()
236 // TODO: Add your control notification handler code here
238 if( this->m_bEditAll
)
239 this->SetAllRebaseAction(CTGitPath::LOGACTIONS_REBASE_EDIT
);
241 this->m_bPickAll
=FALSE
;
242 this->m_bSquashAll
=FALSE
;
243 this->UpdateData(FALSE
);
247 void CRebaseDlg::SetAllRebaseAction(int action
)
249 for(int i
=0;i
<this->m_CommitList
.m_logEntries
.size();i
++)
251 m_CommitList
.m_logEntries
[i
].m_Action
=action
;
253 m_CommitList
.Invalidate();
256 void CRebaseDlg::OnBnClickedRebaseSplit()
259 // TODO: Add your control notification handler code here
262 LRESULT
CRebaseDlg::DefWindowProc(UINT message
, WPARAM wParam
, LPARAM lParam
)
266 if (wParam
== IDC_REBASE_SPLIT
)
268 SPC_NMHDR
* pHdr
= (SPC_NMHDR
*) lParam
;
274 return __super::DefWindowProc(message
, wParam
, lParam
);
277 void CRebaseDlg::DoSize(int delta
)
280 this->RemoveAllAnchors();
282 CSplitterControl::ChangeHeight(GetDlgItem(IDC_COMMIT_LIST
), delta
, CW_TOPALIGN
);
283 //CSplitterControl::ChangeHeight(GetDlgItem(), delta, CW_TOPALIGN);
284 CSplitterControl::ChangeHeight(GetDlgItem(IDC_REBASE_TAB
), -delta
, CW_BOTTOMALIGN
);
285 //CSplitterControl::ChangeHeight(GetDlgItem(), -delta, CW_BOTTOMALIGN);
286 CSplitterControl::ChangePos(GetDlgItem(IDC_SQUASH_ALL
),0,delta
);
287 CSplitterControl::ChangePos(GetDlgItem(IDC_PICK_ALL
),0,delta
);
288 CSplitterControl::ChangePos(GetDlgItem(IDC_EDIT_ALL
),0,delta
);
290 this->AddRebaseAnchor();
291 // adjust the minimum size of the dialog to prevent the resizing from
292 // moving the list control too far down.
294 m_CommitList
.GetClientRect(rcLogMsg
);
295 SetMinTrackSize(CSize(m_DlgOrigRect
.Width(), m_DlgOrigRect
.Height()-m_CommitListOrigRect
.Height()+rcLogMsg
.Height()));
298 // m_CommitList.Invalidate();
300 // GetDlgItem(IDC_LOGMESSAGE)->Invalidate();
302 this->m_ctrlTabCtrl
.Invalidate();
303 this->m_CommitList
.Invalidate();
304 this->m_FileListCtrl
.Invalidate();
305 this->m_LogMessageCtrl
.Invalidate();
309 void CRebaseDlg::SetSplitterRange()
311 if ((m_CommitList
)&&(m_ctrlTabCtrl
))
314 m_CommitList
.GetWindowRect(rcTop
);
315 ScreenToClient(rcTop
);
317 m_ctrlTabCtrl
.GetWindowRect(rcMiddle
);
318 ScreenToClient(rcMiddle
);
319 if (rcMiddle
.Height() && rcMiddle
.Width())
320 m_wndSplitter
.SetRange(rcTop
.top
+60, rcMiddle
.bottom
-80);
324 void CRebaseDlg::OnSize(UINT nType
,int cx
, int cy
)
326 // first, let the resizing take place
327 __super::OnSize(nType
, cx
, cy
);
333 void CRebaseDlg::SaveSplitterPos()
337 CRegDWORD regPos
= CRegDWORD(_T("Software\\TortoiseGit\\TortoiseProc\\ResizableState\\RebaseDlgSizer"));
339 m_wndSplitter
.GetWindowRect(&rectSplitter
);
340 ScreenToClient(&rectSplitter
);
341 regPos
= rectSplitter
.top
;
345 void CRebaseDlg::LoadBranchInfo()
347 m_BranchCtrl
.SetMaxHistoryItems(0x7FFFFFFF);
348 m_UpstreamCtrl
.SetMaxHistoryItems(0x7FFFFFFF);
353 g_Git
.GetBranchList(list
,¤t
,CGit::BRANCH_ALL
);
354 m_BranchCtrl
.AddString(list
);
355 m_UpstreamCtrl
.AddString(list
);
357 m_BranchCtrl
.SetCurSel(current
);
359 AddBranchToolTips(&m_BranchCtrl
);
360 AddBranchToolTips(&m_UpstreamCtrl
);
362 if(!m_Upstream
.IsEmpty())
364 m_UpstreamCtrl
.AddString(m_Upstream
);
365 m_UpstreamCtrl
.SetCurSel(m_UpstreamCtrl
.GetCount()-1);
369 void CRebaseDlg::OnCbnSelchangeBranch()
374 void CRebaseDlg::OnCbnSelchangeUpstream()
379 void CRebaseDlg::FetchLogList()
381 m_CommitList
.Clear();
382 this->m_CommitList
.FillGitLog(NULL
,0,&m_UpstreamCtrl
.GetString(),&m_BranchCtrl
.GetString());
383 if( m_CommitList
.GetItemCount() == 0 )
384 m_CommitList
.ShowText(_T("Nothing Rebase"));
386 CString hash
=g_Git
.GetHash(m_UpstreamCtrl
.GetString());
389 if(m_CommitList
.m_logEntries
[m_CommitList
.m_logEntries
.size()-1].m_ParentHash
.size() >=0 )
391 if(hash
== m_CommitList
.m_logEntries
[m_CommitList
.m_logEntries
.size()-1].m_ParentHash
[0])
393 m_CommitList
.Clear();
394 m_CommitList
.ShowText(_T("Nothing Rebase"));
400 AddBranchToolTips(&this->m_BranchCtrl
);
401 AddBranchToolTips(&this->m_UpstreamCtrl
);
403 for(int i
=0;i
<m_CommitList
.m_logEntries
.size();i
++)
405 m_CommitList
.m_logEntries
[i
].m_Action
= CTGitPath::LOGACTIONS_REBASE_PICK
;
408 m_CommitList
.Invalidate();
410 if(m_CommitList
.m_IsOldFirst
)
411 this->m_CurrentRebaseIndex
= -1;
413 this->m_CurrentRebaseIndex
= m_CommitList
.m_logEntries
.size();
417 void CRebaseDlg::AddBranchToolTips(CHistoryCombo
*pBranch
)
421 CString text
=pBranch
->GetString();
424 g_Git
.GetLog(data
,text
,NULL
,1,0);
426 rev
.ParserFromLog(data
);
427 tooltip
.Format(_T("CommitHash:%s\nCommit by: %s %s\n <b>%s</b> \n %s"),
430 CAppUtils::FormatDateAndTime(rev
.m_AuthorDate
,DATE_LONGDATE
),
434 pBranch
->DisableTooltip();
435 this->m_tooltips
.AddTool(pBranch
->GetComboBoxCtrl(),tooltip
);
439 BOOL
CRebaseDlg::PreTranslateMessage(MSG
*pMsg
)
441 m_tooltips
.RelayEvent(pMsg
);
442 return CResizableStandAloneDialog::PreTranslateMessage(pMsg
);
444 int CRebaseDlg::CheckRebaseCondition()
446 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_LOG
);
448 if( !g_Git
.CheckCleanWorkTree() )
450 CMessageBox::Show(NULL
,_T("Rebase Need Clean Working Tree"),_T("TortoiseGit"),MB_OK
);
453 //Todo Check $REBASE_ROOT
457 cmd
=_T("git.exe var GIT_COMMITTER_IDENT");
458 if(g_Git
.Run(cmd
,NULL
,CP_UTF8
))
461 //Todo call pre_rebase_hook
464 int CRebaseDlg::StartRebase()
468 if(!this->m_IsCherryPick
)
470 //Todo call comment_for_reflog
471 cmd
.Format(_T("git.exe checkout %s"),this->m_BranchCtrl
.GetString());
472 this->AddLogString(cmd
);
474 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
477 this->AddLogString(out
);
480 cmd
=_T("git.exe rev-parse --verify HEAD");
481 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
483 AddLogString(_T("No Head"));
487 //git symbolic-ref HEAD > "$DOTEST"/head-name 2> /dev/null ||
488 // echo "detached HEAD" > "$DOTEST"/head-name
490 cmd
.Format(_T("git.exe update-ref ORIG_HEAD HEAD"));
491 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
493 AddLogString(_T("update ORIG_HEAD Fail"));
497 if( !this->m_IsCherryPick
)
499 cmd
.Format(_T("git.exe checkout %s"),this->m_UpstreamCtrl
.GetString());
500 this->AddLogString(cmd
);
503 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
509 m_OrigUpstreamHash
.Empty();
510 m_OrigUpstreamHash
= g_Git
.GetHash(this->m_UpstreamCtrl
.GetString());
511 if(m_OrigUpstreamHash
.IsEmpty())
513 this->AddLogString(m_OrigUpstreamHash
);
517 if( !this->m_IsCherryPick
)
519 cmd
.Format(_T("git.exe rev-parse %s"),this->m_BranchCtrl
.GetString());
520 if(g_Git
.Run(cmd
,&this->m_OrigBranchHash
,CP_UTF8
))
522 this->AddLogString(m_OrigBranchHash
);
525 this->AddLogString(_T("Start Rebase\r\n"));
528 this->AddLogString(_T("Start Cherry-pick\r\n"));
532 int CRebaseDlg::VerifyNoConflict()
535 if(g_Git
.ListConflictFile(list
))
537 AddLogString(_T("Get conflict files fail"));
540 if( list
.GetCount() != 0 )
542 CMessageBox::Show(NULL
,_T("There are conflict file, you should mark it resolve"),_T("TortoiseGit"),MB_OK
);
548 int CRebaseDlg::FinishRebase()
550 if(this->m_IsCherryPick
) //cherry pick mode no "branch", working at upstream branch
554 cmd
.Format(_T("git.exe branch -f %s"),this->m_BranchCtrl
.GetString());
555 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
561 cmd
.Format(_T("git.exe reset --hard %s"),this->m_OrigUpstreamHash
);
562 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
568 cmd
.Format(_T("git.exe checkout -f %s"),this->m_BranchCtrl
.GetString());
569 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
576 void CRebaseDlg::OnBnClickedContinue()
578 if( m_RebaseStage
== CHOOSE_BRANCH
|| m_RebaseStage
== CHOOSE_COMMIT_PICK_MODE
)
580 if(CheckRebaseCondition())
582 m_RebaseStage
= REBASE_START
;
585 if( m_RebaseStage
== REBASE_DONE
)
590 if( m_RebaseStage
== REBASE_FINISH
)
598 if( m_RebaseStage
== REBASE_SQUASH_CONFLICT
)
600 if(VerifyNoConflict())
602 GitRev
*curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
603 if(this->CheckNextCommitIsSquash())
604 {//next commit is not squash;
605 m_RebaseStage
= REBASE_SQUASH_EDIT
;
606 this->OnRebaseUpdateUI(0,0);
607 this->UpdateCurrentStatus();
611 m_RebaseStage
=REBASE_CONTINUE
;
612 curRev
->m_Action
|=CTGitPath::LOGACTIONS_REBASE_DONE
;
613 this->UpdateCurrentStatus();
617 if( m_RebaseStage
== REBASE_CONFLICT
)
619 if(VerifyNoConflict())
622 GitRev
*curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
626 cmd
.Format(_T("git.exe commit -C %s"), curRev
->m_CommitHash
);
628 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
630 if(!g_Git
.CheckCleanWorkTree())
632 CMessageBox::Show(NULL
,out
,_T("TortoiseGit"),MB_OK
|MB_ICONERROR
);
638 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_LOG
);
639 if( curRev
->m_Action
& CTGitPath::LOGACTIONS_REBASE_EDIT
)
641 m_RebaseStage
=REBASE_EDIT
;
642 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_MESSAGE
);
643 this->UpdateCurrentStatus();
648 m_RebaseStage
=REBASE_CONTINUE
;
649 curRev
->m_Action
|=CTGitPath::LOGACTIONS_REBASE_DONE
;
650 this->UpdateCurrentStatus();
655 if( m_RebaseStage
== REBASE_EDIT
|| m_RebaseStage
== REBASE_SQUASH_EDIT
)
658 GitRev
*curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
660 str
=this->m_LogMessageCtrl
.GetText();
661 if(str
.Trim().IsEmpty())
663 CMessageBox::Show(NULL
,_T("Commit Message Is Empty"),_T("TortoiseGit"),MB_OK
|MB_ICONERROR
);
667 CString tempfile
=::GetTempFile();
668 CFile
file(tempfile
,CFile::modeReadWrite
|CFile::modeCreate
);
669 CStringA log
=CUnicodeUtils::GetUTF8( str
);
670 file
.Write(log
,log
.GetLength());
671 //file.WriteString(m_sLogMessage);
676 if( m_RebaseStage
== REBASE_SQUASH_EDIT
)
677 cmd
.Format(_T("git.exe commit -F \"%s\""), tempfile
);
679 cmd
.Format(_T("git.exe commit --amend -F \"%s\""), tempfile
);
681 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
683 if(!g_Git
.CheckCleanWorkTree())
685 CMessageBox::Show(NULL
,out
,_T("TortoiseGit"),MB_OK
|MB_ICONERROR
);
690 CFile::Remove(tempfile
);
692 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_LOG
);
693 m_RebaseStage
=REBASE_CONTINUE
;
694 curRev
->m_Action
|=CTGitPath::LOGACTIONS_REBASE_DONE
;
695 this->UpdateCurrentStatus();
699 InterlockedExchange(&m_bThreadRunning
, TRUE
);
702 if (AfxBeginThread(RebaseThreadEntry
, this)==NULL
)
704 InterlockedExchange(&m_bThreadRunning
, FALSE
);
705 CMessageBox::Show(NULL
, _T("Create Rebase Thread Fail"), _T("TortoiseGit"), MB_OK
| MB_ICONERROR
);
709 int CRebaseDlg::CheckNextCommitIsSquash()
712 if(m_CommitList
.m_IsOldFirst
)
713 index
=m_CurrentRebaseIndex
+1;
715 index
=m_CurrentRebaseIndex
-1;
722 if(index
>= m_CommitList
.GetItemCount())
725 curRev
=(GitRev
*)m_CommitList
.m_arShownList
[index
];
727 if( curRev
->m_Action
&CTGitPath::LOGACTIONS_REBASE_SQUASH
)
729 if( curRev
->m_Action
&CTGitPath::LOGACTIONS_REBASE_SKIP
)
731 if(m_CommitList
.m_IsOldFirst
)
738 }while(curRev
->m_Action
&CTGitPath::LOGACTIONS_REBASE_SKIP
);
743 int CRebaseDlg::GoNext()
745 if(m_CommitList
.m_IsOldFirst
)
746 m_CurrentRebaseIndex
++;
748 m_CurrentRebaseIndex
--;
752 int CRebaseDlg::StateAction()
754 switch(this->m_RebaseStage
)
757 case CHOOSE_COMMIT_PICK_MODE
:
760 m_RebaseStage
= REBASE_START
;
767 void CRebaseDlg::SetContinueButtonText()
770 switch(this->m_RebaseStage
)
773 case CHOOSE_COMMIT_PICK_MODE
:
778 case REBASE_CONTINUE
:
779 case REBASE_SQUASH_CONFLICT
:
780 Text
= _T("Continue");
783 case REBASE_CONFLICT
:
790 case REBASE_SQUASH_EDIT
:
803 this->GetDlgItem(IDC_REBASE_CONTINUE
)->SetWindowText(Text
);
806 void CRebaseDlg::SetControlEnable()
808 switch(this->m_RebaseStage
)
811 case CHOOSE_COMMIT_PICK_MODE
:
813 this->GetDlgItem(IDC_PICK_ALL
)->EnableWindow(TRUE
);
814 this->GetDlgItem(IDC_EDIT_ALL
)->EnableWindow(TRUE
);
815 this->GetDlgItem(IDC_SQUASH_ALL
)->EnableWindow(TRUE
);
818 this->GetDlgItem(IDC_REBASE_COMBOXEX_BRANCH
)->EnableWindow(TRUE
);
819 this->GetDlgItem(IDC_REBASE_COMBOXEX_UPSTREAM
)->EnableWindow(TRUE
);
821 //this->m_CommitList.m_IsEnableRebaseMenu=TRUE;
822 this->m_CommitList
.m_ContextMenuMask
|= m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_PICK
)|
823 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_SQUASH
)|
824 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_EDIT
)|
825 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_SKIP
);
829 case REBASE_CONTINUE
:
832 case REBASE_CONFLICT
:
834 case REBASE_SQUASH_CONFLICT
:
836 this->GetDlgItem(IDC_PICK_ALL
)->EnableWindow(FALSE
);
837 this->GetDlgItem(IDC_EDIT_ALL
)->EnableWindow(FALSE
);
838 this->GetDlgItem(IDC_SQUASH_ALL
)->EnableWindow(FALSE
);
839 this->GetDlgItem(IDC_REBASE_COMBOXEX_BRANCH
)->EnableWindow(FALSE
);
840 this->GetDlgItem(IDC_REBASE_COMBOXEX_UPSTREAM
)->EnableWindow(FALSE
);
841 //this->m_CommitList.m_IsEnableRebaseMenu=FALSE;
842 this->m_CommitList
.m_ContextMenuMask
&= ~(m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_PICK
)|
843 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_SQUASH
)|
844 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_EDIT
)|
845 m_CommitList
.GetContextMenuBit(CGitLogListBase::ID_REBASE_SKIP
));
851 this->GetDlgItem(IDC_REBASE_CONTINUE
)->EnableWindow(FALSE
);
852 this->GetDlgItem(IDC_REBASE_ABORT
)->EnableWindow(FALSE
);
856 this->GetDlgItem(IDC_REBASE_CONTINUE
)->EnableWindow(TRUE
);
857 this->GetDlgItem(IDC_REBASE_ABORT
)->EnableWindow(TRUE
);
861 void CRebaseDlg::UpdateProgress()
866 if(m_CommitList
.m_IsOldFirst
)
867 index
= m_CurrentRebaseIndex
+1;
869 index
= m_CommitList
.GetItemCount()-m_CurrentRebaseIndex
;
871 m_ProgressBar
.SetRange(1,m_CommitList
.GetItemCount());
872 m_ProgressBar
.SetPos(index
);
874 if(m_CurrentRebaseIndex
>=0 && m_CurrentRebaseIndex
< m_CommitList
.GetItemCount())
877 text
.Format(_T("Rebasing...(%d/%d)"),index
,m_CommitList
.GetItemCount());
878 m_CtrlStatusText
.SetWindowText(text
);
882 GitRev
*prevRev
=NULL
, *curRev
=NULL
;
884 if( m_CurrentRebaseIndex
>= 0 && m_CurrentRebaseIndex
< m_CommitList
.m_arShownList
.GetSize())
886 curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
889 for(int i
=0;i
<m_CommitList
.m_arShownList
.GetSize();i
++)
891 prevRev
=(GitRev
*)m_CommitList
.m_arShownList
[i
];
892 if(prevRev
->m_Action
& CTGitPath::LOGACTIONS_REBASE_CURRENT
)
894 prevRev
->m_Action
&= ~ CTGitPath::LOGACTIONS_REBASE_CURRENT
;
895 m_CommitList
.GetItemRect(i
,&rect
,LVIR_BOUNDS
);
896 m_CommitList
.InvalidateRect(rect
);
902 curRev
->m_Action
|= CTGitPath::LOGACTIONS_REBASE_CURRENT
;
903 m_CommitList
.GetItemRect(m_CurrentRebaseIndex
,&rect
,LVIR_BOUNDS
);
904 m_CommitList
.InvalidateRect(rect
);
906 m_CommitList
.EnsureVisible(m_CurrentRebaseIndex
,FALSE
);
910 void CRebaseDlg::UpdateCurrentStatus()
912 if( m_CurrentRebaseIndex
< 0 && m_RebaseStage
!= REBASE_DONE
)
914 if(m_CommitList
.m_IsOldFirst
)
915 m_RebaseStage
= CRebaseDlg::REBASE_START
;
917 m_RebaseStage
= CRebaseDlg::REBASE_FINISH
;
920 if( m_CurrentRebaseIndex
== m_CommitList
.m_arShownList
.GetSize() && m_RebaseStage
!= REBASE_DONE
)
922 if(m_CommitList
.m_IsOldFirst
)
923 m_RebaseStage
= CRebaseDlg::REBASE_DONE
;
925 m_RebaseStage
= CRebaseDlg::REBASE_FINISH
;
928 SetContinueButtonText();
933 void CRebaseDlg::AddLogString(CString str
)
935 this->m_wndOutputRebase
.SendMessage(SCI_SETREADONLY
, FALSE
);
936 CStringA sTextA
= m_wndOutputRebase
.StringForControl(str
);//CUnicodeUtils::GetUTF8(str);
937 this->m_wndOutputRebase
.SendMessage(SCI_REPLACESEL
, 0, (LPARAM
)(LPCSTR
)sTextA
);
938 this->m_wndOutputRebase
.SendMessage(SCI_REPLACESEL
, 0, (LPARAM
)(LPCSTR
)"\n");
939 this->m_wndOutputRebase
.SendMessage(SCI_SETREADONLY
, TRUE
);
942 int CRebaseDlg::GetCurrentCommitID()
944 if(m_CommitList
.m_IsOldFirst
)
946 return this->m_CurrentRebaseIndex
+1;
950 return m_CommitList
.GetItemCount()-m_CurrentRebaseIndex
;
954 int CRebaseDlg::DoRebase()
957 if(m_CurrentRebaseIndex
<0)
959 if(m_CurrentRebaseIndex
>= m_CommitList
.GetItemCount() )
962 GitRev
*pRev
= (GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
963 int mode
=pRev
->m_Action
& CTGitPath::LOGACTIONS_REBASE_MODE_MASK
;
966 if( mode
== CTGitPath::LOGACTIONS_REBASE_SKIP
)
968 pRev
->m_Action
|= CTGitPath::LOGACTIONS_REBASE_DONE
;
972 if( mode
!= CTGitPath::LOGACTIONS_REBASE_PICK
)
974 this->m_SquashMessage
+= pRev
->m_Subject
;
975 this->m_SquashMessage
+= _T("\n");
976 this->m_SquashMessage
+= pRev
->m_Body
;
979 this->m_SquashMessage
.Empty();
981 if(mode
== CTGitPath::LOGACTIONS_REBASE_SQUASH
)
982 nocommit
=_T(" --no-commit ");
985 log
.Format(_T("%s %d:%s"),CTGitPath::GetActionName(mode
),this->GetCurrentCommitID(),pRev
->m_CommitHash
);
987 AddLogString(pRev
->m_Subject
);
988 cmd
.Format(_T("git.exe cherry-pick %s %s"),nocommit
,pRev
->m_CommitHash
);
990 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
994 if(g_Git
.ListConflictFile(list
))
996 AddLogString(_T("Get conflict files fail"));
999 if(list
.GetCount() == 0 )
1001 if(mode
== CTGitPath::LOGACTIONS_REBASE_PICK
)
1003 pRev
->m_Action
|= CTGitPath::LOGACTIONS_REBASE_DONE
;
1006 if(mode
== CTGitPath::LOGACTIONS_REBASE_EDIT
)
1008 this->m_RebaseStage
= REBASE_EDIT
;
1009 return -1; // Edit return -1 to stop rebase.
1012 if(CheckNextCommitIsSquash())
1014 // let user edit last commmit message
1015 this->m_RebaseStage
= REBASE_SQUASH_EDIT
;
1019 if(mode
== CTGitPath::LOGACTIONS_REBASE_SQUASH
)
1020 m_RebaseStage
= REBASE_SQUASH_CONFLICT
;
1022 m_RebaseStage
= REBASE_CONFLICT
;
1028 if(mode
== CTGitPath::LOGACTIONS_REBASE_PICK
)
1030 pRev
->m_Action
|= CTGitPath::LOGACTIONS_REBASE_DONE
;
1033 if(mode
== CTGitPath::LOGACTIONS_REBASE_EDIT
)
1035 this->m_RebaseStage
= REBASE_EDIT
;
1036 return -1; // Edit return -1 to stop rebase.
1040 if(CheckNextCommitIsSquash())
1042 // let user edit last commmit message
1043 this->m_RebaseStage
= REBASE_SQUASH_EDIT
;
1051 BOOL
CRebaseDlg::IsEnd()
1053 if(m_CommitList
.m_IsOldFirst
)
1054 return m_CurrentRebaseIndex
>= this->m_CommitList
.GetItemCount();
1056 return m_CurrentRebaseIndex
<0;
1059 int CRebaseDlg::RebaseThread()
1064 if( m_RebaseStage
== REBASE_START
)
1066 if( this->StartRebase() )
1068 InterlockedExchange(&m_bThreadRunning
, FALSE
);
1072 m_RebaseStage
= REBASE_CONTINUE
;
1074 }else if( m_RebaseStage
== REBASE_CONTINUE
)
1080 m_RebaseStage
= REBASE_FINISH
;
1092 }else if( m_RebaseStage
== REBASE_FINISH
)
1095 m_RebaseStage
= REBASE_DONE
;
1102 this->PostMessage(MSG_REBASE_UPDATE_UI
);
1103 //this->UpdateCurrentStatus();
1106 InterlockedExchange(&m_bThreadRunning
, FALSE
);
1107 this->PostMessage(MSG_REBASE_UPDATE_UI
);
1111 void CRebaseDlg::ListConflictFile()
1113 this->m_FileListCtrl
.Clear();
1118 this->m_FileListCtrl
.GetStatus(&list
,true);
1119 this->m_FileListCtrl
.Show(CTGitPath::LOGACTIONS_UNMERGED
|CTGitPath::LOGACTIONS_MODIFIED
|CTGitPath::LOGACTIONS_ADDED
|CTGitPath::LOGACTIONS_DELETED
,
1120 CTGitPath::LOGACTIONS_UNMERGED
);
1121 if( this->m_FileListCtrl
.GetItemCount() == 0 )
1127 LRESULT
CRebaseDlg::OnRebaseUpdateUI(WPARAM
,LPARAM
)
1129 UpdateCurrentStatus();
1130 if(m_CurrentRebaseIndex
<0)
1132 if(m_CurrentRebaseIndex
>= m_CommitList
.GetItemCount() )
1134 GitRev
*curRev
=(GitRev
*)m_CommitList
.m_arShownList
[m_CurrentRebaseIndex
];
1136 switch(m_RebaseStage
)
1138 case REBASE_CONFLICT
:
1139 case REBASE_SQUASH_CONFLICT
:
1141 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_CONFLICT
);
1142 this->m_LogMessageCtrl
.SetText(curRev
->m_Subject
+_T("\n")+curRev
->m_Body
);
1145 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_MESSAGE
);
1146 this->m_LogMessageCtrl
.SetText(curRev
->m_Subject
+_T("\n")+curRev
->m_Body
);
1148 case REBASE_SQUASH_EDIT
:
1149 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_MESSAGE
);
1150 this->m_LogMessageCtrl
.SetText(this->m_SquashMessage
);
1153 this->m_ctrlTabCtrl
.SetActiveTab(REBASE_TAB_LOG
);
1157 void CRebaseDlg::OnCancel()
1161 void CRebaseDlg::OnBnClickedAbort()
1164 if(m_OrigUpstreamHash
.IsEmpty())
1166 __super::OnCancel();
1169 if(m_RebaseStage
== CHOOSE_BRANCH
|| m_RebaseStage
== CHOOSE_COMMIT_PICK_MODE
)
1174 if(CMessageBox::Show(NULL
,_T("Are you sure abort rebase"),_T("TortoiseGit"),MB_YESNO
) != IDYES
)
1177 cmd
.Format(_T("git.exe checkout -f %s"),this->m_UpstreamCtrl
.GetString());
1178 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1184 cmd
.Format(_T("git.exe reset --hard %s"),this->m_OrigUpstreamHash
.Left(40));
1185 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1191 if(this->m_IsCherryPick
) //there are not "branch" at cherry pick mode
1194 cmd
.Format(_T("git checkout -f %s"),this->m_BranchCtrl
.GetString());
1195 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1201 cmd
.Format(_T("git.exe reset --hard %s"),this->m_OrigBranchHash
.Left(40));
1202 if(g_Git
.Run(cmd
,&out
,CP_UTF8
))
1207 __super::OnCancel();