using System.Collections.Generic; using System.ComponentModel.DataAnnotations; using System.Threading.Tasks; namespace SourceGit.ViewModels { public class Pull : Popup { public List Remotes => _repo.Remotes; public Models.Branch Current => _current; public bool HasSpecifiedRemoteBranch { get; private set; } public Models.Remote SelectedRemote { get => _selectedRemote; set { if (SetProperty(ref _selectedRemote, value)) PostRemoteSelected(); } } public List RemoteBranches { get => _remoteBranches; private set => SetProperty(ref _remoteBranches, value); } [Required(ErrorMessage = "Remote branch to pull is required!!!")] public Models.Branch SelectedBranch { get => _selectedBranch; set => SetProperty(ref _selectedBranch, value); } public Models.DealWithLocalChanges PreAction { get => _repo.Settings.DealWithLocalChangesOnPull; set => _repo.Settings.DealWithLocalChangesOnPull = value; } public bool UseRebase { get => _repo.Settings.PreferRebaseInsteadOfMerge; set => _repo.Settings.PreferRebaseInsteadOfMerge = value; } public bool FetchAllBranches { get => _repo.Settings.FetchAllBranchesOnPull; set => _repo.Settings.FetchAllBranchesOnPull = value; } public bool NoTags { get => _repo.Settings.FetchWithoutTagsOnPull; set => _repo.Settings.FetchWithoutTagsOnPull = value; } public Pull(Repository repo, Models.Branch specifiedRemoteBranch) { _repo = repo; _current = repo.CurrentBranch; if (specifiedRemoteBranch != null) { _selectedRemote = repo.Remotes.Find(x => x.Name == specifiedRemoteBranch.Remote); _selectedBranch = specifiedRemoteBranch; var branches = new List(); foreach (var branch in _repo.Branches) { if (branch.Remote == specifiedRemoteBranch.Remote) branches.Add(branch); } _remoteBranches = branches; HasSpecifiedRemoteBranch = true; } else { var autoSelectedRemote = null as Models.Remote; if (!string.IsNullOrEmpty(_current.Upstream)) { var remoteNameEndIdx = _current.Upstream.IndexOf('/', 13); if (remoteNameEndIdx > 0) { var remoteName = _current.Upstream.Substring(13, remoteNameEndIdx - 13); autoSelectedRemote = _repo.Remotes.Find(x => x.Name == remoteName); } } if (autoSelectedRemote == null) { var remote = null as Models.Remote; if (!string.IsNullOrEmpty(_repo.Settings.DefaultRemote)) remote = _repo.Remotes.Find(x => x.Name == _repo.Settings.DefaultRemote); _selectedRemote = remote ?? _repo.Remotes[0]; } else { _selectedRemote = autoSelectedRemote; } PostRemoteSelected(); HasSpecifiedRemoteBranch = false; } View = new Views.Pull() { DataContext = this }; } public override Task Sure() { _repo.SetWatcherEnabled(false); return Task.Run(() => { var changes = new Commands.CountLocalChangesWithoutUntracked(_repo.FullPath).Result(); var needPopStash = false; if (changes > 0) { if (PreAction == Models.DealWithLocalChanges.StashAndReaply) { SetProgressDescription("Stash local changes..."); var succ = new Commands.Stash(_repo.FullPath).Push("PULL_AUTO_STASH"); if (!succ) { CallUIThread(() => _repo.SetWatcherEnabled(true)); return false; } needPopStash = true; } else if (PreAction == Models.DealWithLocalChanges.Discard) { SetProgressDescription("Discard local changes ..."); Commands.Discard.All(_repo.FullPath, false); } } bool rs; if (FetchAllBranches) { SetProgressDescription($"Fetching remote: {_selectedRemote.Name}..."); rs = new Commands.Fetch(_repo.FullPath, _selectedRemote.Name, NoTags, SetProgressDescription).Exec(); if (!rs) { CallUIThread(() => _repo.SetWatcherEnabled(true)); return false; } _repo.MarkFetched(); // Use merge/rebase instead of pull as fetch is done manually. if (UseRebase) { SetProgressDescription($"Rebase {_current.Name} on {_selectedBranch.FriendlyName} ..."); rs = new Commands.Rebase(_repo.FullPath, _selectedBranch.FriendlyName, false).Exec(); } else { SetProgressDescription($"Merge {_selectedBranch.FriendlyName} into {_current.Name} ..."); rs = new Commands.Merge(_repo.FullPath, _selectedBranch.FriendlyName, "", SetProgressDescription).Exec(); } } else { SetProgressDescription($"Pull {_selectedRemote.Name}/{_selectedBranch.Name}..."); rs = new Commands.Pull(_repo.FullPath, _selectedRemote.Name, _selectedBranch.Name, UseRebase, NoTags, SetProgressDescription).Exec(); } if (rs && needPopStash) { SetProgressDescription("Re-apply local changes..."); rs = new Commands.Stash(_repo.FullPath).Pop("stash@{0}"); } CallUIThread(() => _repo.SetWatcherEnabled(true)); return rs; }); } private void PostRemoteSelected() { var remoteName = _selectedRemote.Name; var branches = new List(); foreach (var branch in _repo.Branches) { if (branch.Remote == remoteName) branches.Add(branch); } RemoteBranches = branches; var autoSelectedBranch = false; if (!string.IsNullOrEmpty(_current.Upstream) && _current.Upstream.StartsWith($"refs/remotes/{remoteName}/", System.StringComparison.Ordinal)) { foreach (var branch in branches) { if (_current.Upstream == branch.FullName) { SelectedBranch = branch; autoSelectedBranch = true; break; } } } if (!autoSelectedBranch) { foreach (var branch in branches) { if (_current.Name == branch.Name) { SelectedBranch = branch; autoSelectedBranch = true; break; } } } if (!autoSelectedBranch) SelectedBranch = null; } private readonly Repository _repo = null; private readonly Models.Branch _current = null; private Models.Remote _selectedRemote = null; private List _remoteBranches = null; private Models.Branch _selectedBranch = null; } }