/// <summary> /// Loads the view model from octokit models. /// </summary> /// <param name="pullRequest">The pull request model.</param> public async Task Load(PullRequestDetailModel pullRequest) { try { var firstLoad = (Model == null); Model = pullRequest; Author = new ActorViewModel(pullRequest.Author); Title = Resources.PullRequestNavigationItemText + " #" + pullRequest.Number; IsBusy = true; IsFromFork = !pullRequestsService.IsPullRequestFromRepository(LocalRepository, pullRequest); SourceBranchDisplayName = GetBranchDisplayName(IsFromFork, pullRequest.HeadRepositoryOwner, pullRequest.HeadRefName); TargetBranchDisplayName = GetBranchDisplayName(IsFromFork, pullRequest.BaseRepositoryOwner, pullRequest.BaseRefName); Body = !string.IsNullOrWhiteSpace(pullRequest.Body) ? pullRequest.Body : Resources.NoDescriptionProvidedMarkdown; Reviews = PullRequestReviewSummaryViewModel.BuildByUser(Session.User, pullRequest).ToList(); Checks = (IReadOnlyList <IPullRequestCheckViewModel>)PullRequestCheckViewModel.Build(viewViewModelFactory, pullRequest)?.ToList() ?? Array.Empty <IPullRequestCheckViewModel>(); // Only show unresolved comments await Files.InitializeAsync(Session, c => !c.IsResolved); var localBranches = await pullRequestsService.GetLocalBranches(LocalRepository, pullRequest).ToList(); var currentBranch = gitService.GetBranch(LocalRepository); IsCheckedOut = localBranches.Contains(currentBranch); if (IsCheckedOut) { var divergence = await pullRequestsService.CalculateHistoryDivergence(LocalRepository, Model.Number); var pullEnabled = divergence.BehindBy > 0; var pushEnabled = divergence.AheadBy > 0 && !pullEnabled; string pullToolTip; string pushToolTip; if (pullEnabled) { pullToolTip = string.Format( CultureInfo.InvariantCulture, Resources.PullRequestDetailsPullToolTip, IsFromFork ? Resources.Fork : Resources.Remote, SourceBranchDisplayName); } else { pullToolTip = Resources.NoCommitsToPull; } if (pushEnabled) { pushToolTip = string.Format( CultureInfo.InvariantCulture, Resources.PullRequestDetailsPushToolTip, IsFromFork ? Resources.Fork : Resources.Remote, SourceBranchDisplayName); } else if (divergence.AheadBy == 0) { pushToolTip = Resources.NoCommitsToPush; } else { pushToolTip = Resources.MustPullBeforePush; } var submodulesToSync = await pullRequestsService.CountSubmodulesToSync(LocalRepository); var syncSubmodulesToolTip = string.Format(CultureInfo.InvariantCulture, Resources.SyncSubmodules, submodulesToSync); UpdateState = new UpdateCommandState(divergence, pullEnabled, pushEnabled, pullToolTip, pushToolTip, syncSubmodulesToolTip, submodulesToSync); CheckoutState = null; } else { var caption = localBranches.Count > 0 ? string.Format( CultureInfo.InvariantCulture, Resources.PullRequestDetailsCheckout, localBranches.First().DisplayName) : string.Format( CultureInfo.InvariantCulture, Resources.PullRequestDetailsCheckoutTo, await pullRequestsService.GetDefaultLocalBranchName(LocalRepository, Model.Number, Model.Title)); var clean = await pullRequestsService.IsWorkingDirectoryClean(LocalRepository); string disabled = null; if (pullRequest.HeadRepositoryOwner == null) { disabled = Resources.SourceRepositoryNoLongerAvailable; } else if (!clean) { disabled = Resources.WorkingDirectoryHasUncommittedCHanges; } CheckoutState = new CheckoutCommandState(caption, disabled); UpdateState = null; } sessionSubscription?.Dispose(); sessionSubscription = Session.WhenAnyValue(x => x.HasPendingReview) .Skip(1) .Subscribe(x => Reviews = PullRequestReviewSummaryViewModel.BuildByUser(Session.User, Session.PullRequest).ToList()); if (firstLoad) { usageTracker.IncrementCounter(x => x.NumberOfPullRequestsOpened).Forget(); } if (!isInCheckout) { pullRequestsService.RemoveUnusedRemotes(LocalRepository).Subscribe(_ => { }); } } finally { IsBusy = false; } }
/// <summary> /// Loads the view model from octokit models. /// </summary> /// <param name="pullRequest">The pull request model.</param> /// <param name="files">The pull request's changed files.</param> public async Task Load(IPullRequestModel pullRequest) { Model = pullRequest; SourceBranchDisplayName = GetBranchDisplayName(pullRequest.Head?.Label); TargetBranchDisplayName = GetBranchDisplayName(pullRequest.Base.Label); Body = !string.IsNullOrWhiteSpace(pullRequest.Body) ? pullRequest.Body : "*No description provided.*"; ChangedFilesTree.Clear(); ChangedFilesList.Clear(); // WPF doesn't support AddRange here so iterate through the changes. foreach (var change in CreateChangedFilesList(pullRequest.ChangedFiles)) { ChangedFilesList.Add(change); } foreach (var change in CreateChangedFilesTree(ChangedFilesList).Children) { ChangedFilesTree.Add(change); } var localBranches = await pullRequestsService.GetLocalBranches(repository, pullRequest).ToList(); if (localBranches.Contains(repository.CurrentBranch)) { var divergence = await pullRequestsService.CalculateHistoryDivergence(repository, Model.Number); if (divergence.BehindBy == null) { CheckoutMode = CheckoutMode.InvalidState; } else if (divergence.AheadBy > 0) { CheckoutMode = pullRequestsService.IsPullRequestFromFork(repository, pullRequest) ? CheckoutMode.InvalidState : CheckoutMode.Push; } else if (divergence.BehindBy == 0) { CheckoutMode = CheckoutMode.UpToDate; } else { CheckoutMode = CheckoutMode.NeedsPull; CommitsBehind = divergence.BehindBy.Value; } } else if (localBranches.Count > 0) { CheckoutMode = CheckoutMode.Switch; } else { CheckoutMode = CheckoutMode.Fetch; } var clean = await pullRequestsService.IsCleanForCheckout(repository); CheckoutDisabledMessage = (!clean && CheckoutMode != CheckoutMode.UpToDate && CheckoutMode != CheckoutMode.Push) ? $"Cannot {GetCheckoutModeDescription(CheckoutMode)} as your working directory has uncommitted changes." : null; IsBusy = false; }