File: Preview\SolutionPreviewResult.cs
Web Access
Project: src\src\EditorFeatures\Core\Microsoft.CodeAnalysis.EditorFeatures.csproj (Microsoft.CodeAnalysis.EditorFeatures)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.Editor.Shared.Extensions;
using Microsoft.CodeAnalysis.Editor.Shared.Utilities;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.Editor;
 
internal class SolutionPreviewResult(
    IThreadingContext threadingContext,
    IList<SolutionPreviewItem>? previews,
    SolutionChangeSummary? changeSummary = null)
{
    private readonly IThreadingContext _threadingContext = threadingContext;
    private readonly IList<SolutionPreviewItem> _previews = previews ?? SpecializedCollections.EmptyList<SolutionPreviewItem>();
    public readonly SolutionChangeSummary? ChangeSummary = changeSummary;
 
    public SolutionPreviewResult(IThreadingContext threadingContext, SolutionPreviewItem preview, SolutionChangeSummary? changeSummary = null)
        : this(threadingContext, [preview], changeSummary)
    {
    }
 
    public bool IsEmpty => _previews.Count == 0;
 
    public async Task<IReadOnlyList<object>?> GetPreviewsAsync(DocumentId? preferredDocumentId = null, ProjectId? preferredProjectId = null, CancellationToken cancellationToken = default)
    {
        _threadingContext.ThrowIfNotOnUIThread();
        cancellationToken.ThrowIfCancellationRequested();
 
        var orderedPreviews = _previews.OrderBy((i1, i2) =>
        {
            return i1.DocumentId == preferredDocumentId && i2.DocumentId != preferredDocumentId ? -1 :
                   i1.DocumentId != preferredDocumentId && i2.DocumentId == preferredDocumentId ? 1 :
                   _previews.IndexOf(i1) - _previews.IndexOf(i2);
        }).ThenBy((i1, i2) =>
        {
            return i1.ProjectId == preferredProjectId && i2.ProjectId != preferredProjectId ? -1 :
                   i1.ProjectId != preferredProjectId && i2.ProjectId == preferredProjectId ? 1 :
                   _previews.IndexOf(i1) - _previews.IndexOf(i2);
        }).ThenBy((i1, i2) =>
        {
            return i1.Text == null && i2.Text != null ? -1 :
                   i1.Text != null && i2.Text == null ? 1 :
                   _previews.IndexOf(i1) - _previews.IndexOf(i2);
        });
 
        var result = new List<object>();
        var gotRichPreview = false;
 
        try
        {
            foreach (var previewItem in _previews)
            {
                cancellationToken.ThrowIfCancellationRequested();
                if (previewItem.Text != null)
                {
                    result.Add(previewItem.Text);
                }
                else if (!gotRichPreview)
                {
                    var preview = await previewItem.LazyPreview(cancellationToken).ConfigureAwait(true);
                    if (preview != null)
                    {
                        result.Add(preview);
                        gotRichPreview = true;
                    }
                }
            }
 
            return result.Count == 0 ? null : result;
        }
        catch (OperationCanceledException)
        {
            // make sure we dispose all disposable preview objects before
            // we let control to exit this method
            result.OfType<IDisposable>().Do(d => d.Dispose());
            throw;
        }
    }
 
    /// <summary>Merge two different previews into one final preview result.  The final preview will
    /// have a concatenation of all the inidivual previews contained within each result.</summary>
    internal static SolutionPreviewResult? Merge(SolutionPreviewResult? result1, SolutionPreviewResult? result2)
    {
        if (result1 == null)
        {
            return result2;
        }
 
        if (result2 == null)
        {
            return result1;
        }
 
        return new SolutionPreviewResult(
            result1._threadingContext,
            [.. result1._previews, .. result2._previews],
            result1.ChangeSummary ?? result2.ChangeSummary);
    }
}