1 // TortoiseGit - a Windows shell extension for easy version control
3 // Copyright (C) 2008-2016 - 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 // SettingGitRemote.cpp : implementation file
23 #include "TortoiseProc.h"
24 #include "SettingGitRemote.h"
26 #include "GitAdminDir.h"
27 #include "MessageBox.h"
30 #include "HistoryCombo.h"
32 // CSettingGitRemote dialog
34 IMPLEMENT_DYNAMIC(CSettingGitRemote
, ISettingsPropPage
)
36 CSettingGitRemote::CSettingGitRemote()
37 : ISettingsPropPage(CSettingGitRemote::IDD
)
41 , m_bPushDefault(FALSE
)
46 CSettingGitRemote::~CSettingGitRemote()
50 void CSettingGitRemote::DoDataExchange(CDataExchange
* pDX
)
52 CPropertyPage::DoDataExchange(pDX
);
53 DDX_Control(pDX
, IDC_LIST_REMOTE
, m_ctrlRemoteList
);
54 DDX_Text(pDX
, IDC_EDIT_REMOTE
, m_strRemote
);
55 DDX_Text(pDX
, IDC_EDIT_URL
, m_strUrl
);
56 DDX_Text(pDX
, IDC_EDIT_PUSHURL
, m_strPushUrl
);
57 DDX_Text(pDX
, IDC_EDIT_PUTTY_KEY
, m_strPuttyKeyfile
);
58 DDX_Control(pDX
, IDC_COMBO_TAGOPT
, m_ctrlTagOpt
);
59 DDX_Check(pDX
, IDC_CHECK_PRUNE
, m_bPrune
);
60 DDX_Check(pDX
, IDC_CHECK_PRUNEALL
, m_bPruneAll
);
61 DDX_Check(pDX
, IDC_CHECK_PUSHDEFAULT
, m_bPushDefault
);
64 BEGIN_MESSAGE_MAP(CSettingGitRemote
, CPropertyPage
)
66 ON_BN_CLICKED(IDC_BUTTON_BROWSE
, &CSettingGitRemote::OnBnClickedButtonBrowse
)
67 ON_BN_CLICKED(IDC_BUTTON_ADD
, &CSettingGitRemote::OnBnClickedButtonAdd
)
68 ON_LBN_SELCHANGE(IDC_LIST_REMOTE
, &CSettingGitRemote::OnLbnSelchangeListRemote
)
69 ON_EN_CHANGE(IDC_EDIT_REMOTE
, &CSettingGitRemote::OnEnChangeEditRemote
)
70 ON_EN_CHANGE(IDC_EDIT_URL
, &CSettingGitRemote::OnEnChangeEditUrl
)
71 ON_EN_CHANGE(IDC_EDIT_PUSHURL
, &CSettingGitRemote::OnEnChangeEditPushUrl
)
72 ON_EN_CHANGE(IDC_EDIT_PUTTY_KEY
, &CSettingGitRemote::OnEnChangeEditPuttyKey
)
73 ON_CBN_SELCHANGE(IDC_COMBO_TAGOPT
, &CSettingGitRemote::OnCbnSelchangeComboTagOpt
)
74 ON_BN_CLICKED(IDC_CHECK_PRUNE
, &CSettingGitRemote::OnBnClickedCheckprune
)
75 ON_BN_CLICKED(IDC_CHECK_PRUNEALL
, &CSettingGitRemote::OnBnClickedCheckpruneall
)
76 ON_BN_CLICKED(IDC_CHECK_PUSHDEFAULT
, &CSettingGitRemote::OnBnClickedCheckpushdefault
)
77 ON_BN_CLICKED(IDC_BUTTON_REMOVE
, &CSettingGitRemote::OnBnClickedButtonRemove
)
78 ON_BN_CLICKED(IDC_BUTTON_RENAME_REMOTE
, &CSettingGitRemote::OnBnClickedButtonRenameRemote
)
81 static void ShowEditBalloon(HWND hDlg
, UINT nIdControl
, UINT nIdText
, UINT nIdTitle
, int nIcon
= TTI_WARNING
)
83 CString
text(MAKEINTRESOURCE(nIdText
));
84 CString
title(MAKEINTRESOURCE(nIdTitle
));
86 bt
.cbStruct
= sizeof(bt
);
90 SendDlgItemMessage(hDlg
, nIdControl
, EM_SHOWBALLOONTIP
, 0, (LPARAM
)&bt
);
93 #define TIMER_PREFILL 1
95 BOOL
CSettingGitRemote::OnInitDialog()
97 ISettingsPropPage::OnInitDialog();
99 AdjustControlSize(IDC_CHECK_PRUNE
);
100 AdjustControlSize(IDC_CHECK_PRUNEALL
);
101 AdjustControlSize(IDC_CHECK_PUSHDEFAULT
);
103 STRING_VECTOR remotes
;
104 g_Git
.GetRemoteList(remotes
);
105 m_ctrlRemoteList
.ResetContent();
106 for (size_t i
= 0; i
< remotes
.size(); i
++)
107 m_ctrlRemoteList
.AddString(remotes
[i
]);
109 m_ctrlTagOpt
.AddString(CString(MAKEINTRESOURCE(IDS_FETCH_REACHABLE
)));
110 m_ctrlTagOpt
.AddString(CString(MAKEINTRESOURCE(IDS_NONE
)));
111 m_ctrlTagOpt
.AddString(CString(MAKEINTRESOURCE(IDS_ALL
)));
112 m_ctrlTagOpt
.SetCurSel(0);
114 CString pruneAll
= g_Git
.GetConfigValue(_T("fetch.prune"));
115 m_bPruneAll
= pruneAll
== _T("true") ? TRUE
: FALSE
;
117 //this->GetDlgItem(IDC_EDIT_REMOTE)->EnableWindow(FALSE);
118 this->UpdateData(FALSE
);
120 SetTimer(TIMER_PREFILL
, 1000, nullptr);
123 // CSettingGitRemote message handlers
125 void CSettingGitRemote::OnTimer(UINT_PTR nIDEvent
)
127 if (nIDEvent
== TIMER_PREFILL
)
129 if (m_strRemote
.IsEmpty() && m_ctrlRemoteList
.GetCount() == 0)
130 ShowEditBalloon(IDC_EDIT_URL
, IDS_B_T_PREFILL_ORIGIN
, IDS_HINT
, TTI_INFO
);
132 KillTimer(TIMER_PREFILL
);
136 void CSettingGitRemote::OnBnClickedButtonBrowse()
138 CFileDialog
dlg(TRUE
, nullptr, nullptr, OFN_HIDEREADONLY
| OFN_OVERWRITEPROMPT
, CString(MAKEINTRESOURCE(IDS_PUTTYKEYFILEFILTER
)));
141 INT_PTR ret
= dlg
.DoModal();
142 SetCurrentDirectory(g_Git
.m_CurrentDir
);
145 this->m_strPuttyKeyfile
= dlg
.GetPathName();
146 this->UpdateData(FALSE
);
147 OnEnChangeEditPuttyKey();
151 void CSettingGitRemote::OnBnClickedButtonAdd()
155 if(m_strRemote
.IsEmpty())
157 CMessageBox::Show(GetSafeHwnd(), IDS_PROC_GITCONFIG_REMOTEEMPTY
, IDS_APPNAME
, MB_OK
| MB_ICONERROR
);
160 if(m_strUrl
.IsEmpty())
162 CMessageBox::Show(GetSafeHwnd(), IDS_PROC_GITCONFIG_URLEMPTY
, IDS_APPNAME
, MB_OK
| MB_ICONERROR
);
166 m_ChangedMask
= REMOTE_NAME
| REMOTE_URL
| REMOTE_PUTTYKEY
| REMOTE_TAGOPT
| REMOTE_PRUNE
| REMOTE_PRUNEALL
| REMOTE_PUSHDEFAULT
| REMOTE_PUSHURL
;
167 if(IsRemoteExist(m_strRemote
))
170 msg
.Format(IDS_PROC_GITCONFIG_OVERWRITEREMOTE
, (LPCTSTR
)m_strRemote
);
171 if (CMessageBox::Show(GetSafeHwnd(), msg
, L
"TortoiseGit", MB_YESNO
| MB_ICONQUESTION
| MB_DEFBUTTON2
) == IDYES
)
172 m_ChangedMask
&= ~REMOTE_NAME
;
180 BOOL
CSettingGitRemote::IsRemoteExist(CString
&remote
)
183 for(int i
=0;i
<m_ctrlRemoteList
.GetCount();i
++)
185 m_ctrlRemoteList
.GetText(i
,str
);
192 struct CheckRefspecStruct
198 static int CheckRemoteCollideWithRefspec(const git_config_entry
*entry
, void * payload
)
200 auto crs
= reinterpret_cast<CheckRefspecStruct
*>(payload
);
202 if (entry
->name
== "remote." + crs
->remote
+ ".fetch")
204 CStringA value
= CStringA(entry
->value
);
205 CStringA match
= ":refs/remotes/" + crs
->remote
;
206 int pos
= value
.Find(match
);
209 if ((pos
+ match
.GetLength() == value
.GetLength()) || (value
[pos
+ match
.GetLength()] == '/'))
217 bool CSettingGitRemote::IsRemoteCollideWithRefspec(CString remote
)
219 CheckRefspecStruct crs
= { CUnicodeUtils::GetUTF8(remote
), false };
220 CAutoConfig
config(true);
221 git_config_add_file_ondisk(config
, CGit::GetGitPathStringA(g_Git
.GetGitLocalConfig()), GIT_CONFIG_LEVEL_LOCAL
, FALSE
);
222 for (const auto pattern
: { "remote\\..*\\.fetch", "svn-remote\\..*\\.fetch", "svn-remote\\..*\\.branches", "svn-remote\\..*\\.tags" })
224 git_config_foreach_match(config
, pattern
, CheckRemoteCollideWithRefspec
, &crs
);
231 void CSettingGitRemote::OnLbnSelchangeListRemote()
237 if (CMessageBox::Show(GetSafeHwnd(), IDS_PROC_GITCONFIG_SAVEREMOTE
, IDS_APPNAME
, 1, IDI_QUESTION
, IDS_SAVEBUTTON
, IDS_DISCARDBUTTON
) == 1)
244 index
= this->m_ctrlRemoteList
.GetCurSel();
248 m_strPushUrl
.Empty();
250 m_strPuttyKeyfile
.Empty();
251 this->UpdateData(FALSE
);
255 m_ctrlRemoteList
.GetText(index
,remote
);
256 this->m_strRemote
=remote
;
258 cmd
.Format(_T("remote.%s.url"), (LPCTSTR
)remote
);
260 m_strUrl
= g_Git
.GetConfigValue(cmd
);
262 cmd
.Format(_T("remote.%s.pushurl"), (LPCTSTR
)remote
);
263 m_strPushUrl
.Empty();
264 m_strPushUrl
= g_Git
.GetConfigValue(cmd
);
266 cmd
.Format(_T("remote.%s.puttykeyfile"), (LPCTSTR
)remote
);
268 this->m_strPuttyKeyfile
= g_Git
.GetConfigValue(cmd
);
273 cmd
.Format(_T("remote.%s.tagopt"), (LPCTSTR
)remote
);
274 CString tagopt
= g_Git
.GetConfigValue(cmd
);
276 if (tagopt
== "--no-tags")
278 else if (tagopt
== "--tags")
280 m_ctrlTagOpt
.SetCurSel(index
);
282 CString pushDefault
= g_Git
.GetConfigValue(_T("remote.pushdefault"));
283 m_bPushDefault
= pushDefault
== remote
? TRUE
: FALSE
;
284 cmd
.Format(_T("remote.%s.prune"), (LPCTSTR
)remote
);
285 CString prune
= g_Git
.GetConfigValue(cmd
);
286 m_bPrune
= prune
== _T("true") ? TRUE
: prune
== _T("false") ? FALSE
: 2;
287 CString pruneAll
= g_Git
.GetConfigValue(_T("fetch.prune"));
288 m_bPruneAll
= pruneAll
== _T("true") ? TRUE
: FALSE
;
290 GetDlgItem(IDC_BUTTON_ADD
)->EnableWindow(TRUE
);
291 GetDlgItem(IDC_BUTTON_REMOVE
)->EnableWindow(TRUE
);
292 GetDlgItem(IDC_BUTTON_RENAME_REMOTE
)->EnableWindow(TRUE
);
293 this->UpdateData(FALSE
);
296 void CSettingGitRemote::OnEnChangeEditRemote()
298 m_ChangedMask
|=REMOTE_NAME
;
302 if (IsRemoteCollideWithRefspec(m_strRemote
))
303 ShowEditBalloon(IDC_EDIT_REMOTE
, IDS_B_T_REMOTE_NAME_COLLIDE
, IDS_HINT
, TTI_WARNING
);
304 if( (!this->m_strRemote
.IsEmpty())&&(!this->m_strUrl
.IsEmpty()) )
308 void CSettingGitRemote::OnEnChangeEditUrl()
310 m_ChangedMask
|=REMOTE_URL
;
314 if (m_strRemote
.IsEmpty() && !m_strUrl
.IsEmpty() && m_ctrlRemoteList
.GetCount() == 0)
316 GetDlgItem(IDC_EDIT_REMOTE
)->SetWindowText(_T("origin"));
317 OnEnChangeEditRemote();
320 if( (!this->m_strRemote
.IsEmpty())&&(!this->m_strUrl
.IsEmpty()) )
324 void CSettingGitRemote::OnEnChangeEditPushUrl()
326 m_ChangedMask
|= REMOTE_PUSHURL
;
330 if (!this->m_strRemote
.IsEmpty())
334 void CSettingGitRemote::OnEnChangeEditPuttyKey()
336 m_ChangedMask
|=REMOTE_PUTTYKEY
;
339 if (!this->m_strUrl
.IsEmpty())
343 void CSettingGitRemote::OnCbnSelchangeComboTagOpt()
345 m_ChangedMask
|= REMOTE_TAGOPT
;
351 void CSettingGitRemote::OnBnClickedCheckprune()
353 m_ChangedMask
|= REMOTE_PRUNE
;
359 void CSettingGitRemote::OnBnClickedCheckpruneall()
361 m_ChangedMask
|= REMOTE_PRUNEALL
;
366 void CSettingGitRemote::OnBnClickedCheckpushdefault()
368 m_ChangedMask
|= REMOTE_PUSHDEFAULT
;
373 BOOL
CSettingGitRemote::Save(CString key
,CString value
)
377 cmd
.Format(_T("remote.%s.%s"), (LPCTSTR
)m_strRemote
, (LPCTSTR
)key
);
380 // don't check result code. it fails if the entry not exist
381 g_Git
.UnsetConfigValue(cmd
, CONFIG_LOCAL
);
382 if (!g_Git
.GetConfigValue(cmd
).IsEmpty())
385 msg
.Format(IDS_PROC_SAVECONFIGFAILED
, (LPCTSTR
)cmd
, (LPCTSTR
)value
);
386 CMessageBox::Show(GetSafeHwnd(), msg
, L
"TortoiseGit", MB_OK
| MB_ICONERROR
);
392 if (g_Git
.SetConfigValue(cmd
, value
, CONFIG_LOCAL
))
395 msg
.Format(IDS_PROC_SAVECONFIGFAILED
, (LPCTSTR
)cmd
, (LPCTSTR
)value
);
396 CMessageBox::Show(GetSafeHwnd(), msg
, L
"TortoiseGit", MB_OK
| MB_ICONERROR
);
402 BOOL
CSettingGitRemote::SaveGeneral(CString key
, CString value
)
406 // don't check result code. it fails if the entry not exist
407 g_Git
.UnsetConfigValue(key
, CONFIG_LOCAL
);
408 if (!g_Git
.GetConfigValue(key
).IsEmpty())
411 msg
.Format(IDS_PROC_SAVECONFIGFAILED
, (LPCTSTR
)key
, (LPCTSTR
)value
);
412 CMessageBox::Show(GetSafeHwnd(), msg
, L
"TortoiseGit", MB_OK
| MB_ICONERROR
);
418 if (g_Git
.SetConfigValue(key
, value
, CONFIG_LOCAL
))
421 msg
.Format(IDS_PROC_SAVECONFIGFAILED
, (LPCTSTR
)key
, (LPCTSTR
)value
);
422 CMessageBox::Show(GetSafeHwnd(), msg
, L
"TortoiseGit", MB_OK
| MB_ICONERROR
);
429 BOOL
CSettingGitRemote::OnApply()
434 if (m_ChangedMask
& REMOTE_PUSHDEFAULT
)
436 if (!m_strRemote
.Trim().IsEmpty() && m_bPushDefault
)
438 if (!SaveGeneral(_T("remote.pushdefault"), m_strRemote
.Trim()))
443 if (!SaveGeneral(_T("remote.pushdefault"), _T("")))
447 m_ChangedMask
&= ~REMOTE_PUSHDEFAULT
;
450 if (m_ChangedMask
& REMOTE_PRUNEALL
)
452 if (!SaveGeneral(_T("fetch.prune"), m_bPruneAll
== TRUE
? _T("true") : _T("")))
454 m_ChangedMask
&= ~REMOTE_PRUNEALL
;
457 if (m_ChangedMask
&& m_strRemote
.Trim().IsEmpty())
459 CMessageBox::Show(GetSafeHwnd(), IDS_PROC_GITCONFIG_REMOTEEMPTY
, IDS_APPNAME
, MB_OK
| MB_ICONERROR
);
463 if(m_ChangedMask
& REMOTE_NAME
)
466 if(m_strRemote
.IsEmpty())
468 CMessageBox::Show(GetSafeHwnd(), IDS_PROC_GITCONFIG_REMOTEEMPTY
, IDS_APPNAME
, MB_OK
| MB_ICONERROR
);
471 if(m_strUrl
.IsEmpty())
473 CMessageBox::Show(GetSafeHwnd(), IDS_PROC_GITCONFIG_URLEMPTY
, IDS_APPNAME
, MB_OK
| MB_ICONERROR
);
477 if (m_ctrlRemoteList
.GetCount() > 0)
479 // tagopt not --no-tags
480 if (m_ctrlTagOpt
.GetCurSel() != 1)
482 if (CMessageBox::ShowCheck(GetSafeHwnd(), IDS_PROC_GITCONFIG_ASKTAGOPT
, IDS_APPNAME
, MB_YESNO
| MB_ICONQUESTION
, _T("TagOptNoTagsWarning"), IDS_MSGBOX_DONOTSHOWAGAIN
) == IDYES
)
484 m_ctrlTagOpt
.SetCurSel(1);
485 m_ChangedMask
|= REMOTE_TAGOPT
;
490 m_strUrl
.Replace(L
'\\', L
'/');
492 cmd
.Format(_T("git.exe remote add \"%s\" \"%s\""), (LPCTSTR
)m_strRemote
, (LPCTSTR
)m_strUrl
);
493 if (g_Git
.Run(cmd
, &out
, CP_UTF8
))
495 CMessageBox::Show(GetSafeHwnd(), out
, L
"TorotiseGit", MB_OK
| MB_ICONERROR
);
498 m_ChangedMask
&= ~REMOTE_URL
;
500 m_ctrlRemoteList
.SetCurSel(m_ctrlRemoteList
.AddString(m_strRemote
));
501 GetDlgItem(IDC_BUTTON_ADD
)->EnableWindow(TRUE
);
502 GetDlgItem(IDC_BUTTON_RENAME_REMOTE
)->EnableWindow(TRUE
);
503 if (!m_bNoFetch
&& CMessageBox::Show(GetSafeHwnd(), IDS_SETTINGS_FETCH_ADDEDREMOTE
, IDS_APPNAME
, MB_ICONQUESTION
| MB_YESNO
) == IDYES
)
504 CCommonAppUtils::RunTortoiseGitProc(_T("/command:fetch /path:\"") + g_Git
.m_CurrentDir
+ _T("\" /remote:\"") + m_strRemote
+ _T("\""));
506 if(m_ChangedMask
& REMOTE_URL
)
508 m_strUrl
.Replace(L
'\\', L
'/');
509 if (!Save(_T("url"),this->m_strUrl
))
513 if(m_ChangedMask
& REMOTE_PUTTYKEY
)
515 if (!Save(_T("puttykeyfile"),this->m_strPuttyKeyfile
))
519 if (m_ChangedMask
& REMOTE_TAGOPT
)
522 int index
= m_ctrlTagOpt
.GetCurSel();
524 tagopt
= "--no-tags";
527 if (!Save(_T("tagopt"), tagopt
))
531 if (m_ChangedMask
& REMOTE_PRUNE
)
533 if (!Save(_T("prune"), m_bPrune
== TRUE
? _T("true") : m_bPrune
== FALSE
? _T("false") : _T("")))
537 if (m_ChangedMask
& REMOTE_PUSHURL
)
539 m_strPushUrl
.Replace(L
'\\', L
'/');
540 if (!Save(_T("pushurl"), m_strPushUrl
))
547 return ISettingsPropPage::OnApply();
550 void CleanupSyncRemotes(const CString
& remote
)
552 CString workingDir
= g_Git
.m_CurrentDir
;
553 workingDir
.Replace(_T(':'), _T('_'));
554 CHistoryCombo::RemoveEntryFromHistory(_T("Software\\TortoiseGit\\History\\SyncURL\\") + workingDir
, _T("url"), remote
);
557 void CSettingGitRemote::OnBnClickedButtonRemove()
560 index
=m_ctrlRemoteList
.GetCurSel();
564 m_ctrlRemoteList
.GetText(index
,str
);
566 msg
.Format(IDS_WARN_REMOVE
, (LPCTSTR
)str
);
567 if (CMessageBox::Show(GetSafeHwnd(), msg
, L
"TortoiseGit", MB_YESNO
| MB_ICONQUESTION
) == IDYES
)
570 cmd
.Format(_T("git.exe remote rm %s"), (LPCTSTR
)str
);
571 if (g_Git
.Run(cmd
, &out
, CP_UTF8
))
573 CMessageBox::Show(GetSafeHwnd(), out
, L
"TortoiseGit", MB_OK
| MB_ICONERROR
);
577 CleanupSyncRemotes(str
);
578 m_ctrlRemoteList
.DeleteString(index
);
579 OnLbnSelchangeListRemote();
584 void CSettingGitRemote::OnBnClickedButtonRenameRemote()
586 int sel
= m_ctrlRemoteList
.GetCurSel();
589 CString oldRemote
, newRemote
;
590 m_ctrlRemoteList
.GetText(sel
, oldRemote
);
591 GetDlgItem(IDC_EDIT_REMOTE
)->GetWindowText(newRemote
);
593 cmd
.Format(_T("git.exe remote rename %s %s"), (LPCTSTR
)oldRemote
, (LPCTSTR
)newRemote
);
594 if (g_Git
.Run(cmd
, &out
, CP_UTF8
))
596 CMessageBox::Show(GetSafeHwnd(), out
, L
"TortoiseGit", MB_OK
| MB_ICONERROR
);
600 CleanupSyncRemotes(oldRemote
);
601 m_ctrlRemoteList
.DeleteString(sel
);
602 m_ctrlRemoteList
.SetCurSel(m_ctrlRemoteList
.AddString(newRemote
));
603 m_ChangedMask
&= ~REMOTE_NAME
;
605 this->SetModified(FALSE
);