File: Completion\Providers\AbstractPreprocessorCompletionProvider.cs
Web Access
Project: ..\..\..\src\Features\Core\Portable\Microsoft.CodeAnalysis.Features.csproj (Microsoft.CodeAnalysis.Features)
// 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.Collections.Generic;
using System.Collections.Immutable;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.LanguageService;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Microsoft.CodeAnalysis.Shared.Extensions.ContextQuery;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.Completion.Providers
{
    internal abstract class AbstractPreprocessorCompletionProvider : LSPCompletionProvider
    {
        public sealed override async Task ProvideCompletionsAsync(CompletionContext context)
        {
            var cancellationToken = context.CancellationToken;
            var originatingDocument = context.Document;
            var position = context.Position;
            var solution = originatingDocument.Project.Solution;
            var syntaxContext = await context.GetSyntaxContextWithExistingSpeculativeModelAsync(originatingDocument, cancellationToken).ConfigureAwait(false);
            if (!syntaxContext.IsPreProcessorExpressionContext)
                return;
 
            // Walk all the projects this document is linked in so that we get the full set of preprocessor symbols
            // defined across all of them.
            var syntaxFacts = originatingDocument.GetRequiredLanguageService<ISyntaxFactsService>();
            var preprocessorNames = new HashSet<string>(syntaxFacts.StringComparer);
 
            foreach (var documentId in solution.GetRelatedDocumentIds(originatingDocument.Id))
            {
                var document = solution.GetRequiredDocument(documentId);
                var currentSyntaxTree = await document.GetRequiredSyntaxTreeAsync(cancellationToken).ConfigureAwait(false);
 
                preprocessorNames.AddRange(currentSyntaxTree.Options.PreprocessorSymbolNames);
            }
 
            // Keep all the preprocessor symbol names together.  We don't want to intermingle them with any keywords we
            // include (like `true/false`)
            foreach (var name in preprocessorNames.OrderBy(a => a))
            {
                context.AddItem(CommonCompletionItem.Create(
                    name,
                    displayTextSuffix: "",
                    CompletionItemRules.Default,
                    glyph: Glyph.Keyword,
                    sortText: "_0_" + name));
            }
        }
    }
}